PWA(Progressivewebapps,渐进式Web应用)使用现代的WebAPI以及传统的渐进式增强策略来创建跨平台Web应用程序,说白了,PWA可以让我们的站点以原生APP的形式运行,但相比于安装原生APP应用,访问PWA显然更加容易和迅速,还可以通过链接来分享PWA应用。 有许多知名的网络平台已经将PWA方案落地,比如Twitter。选择增强的网站体验而不是原生应用。事实上使用PWA也确实从中获得了显而易见的益处。https:www。pwastats。com这个网站上分享了许多案例研究,PWA相比于传统应用有以下好处: 1、减少应用安装后的加载时间,通过ServiceWorkers来进行缓存,以此来节省带宽和时间。 2、当应用有可用的更新时,可以只更新发生改变的那部分内容。相比之下,对于一个原生应用而言,即便是最微小的改动也需要强制用户去进行热更新或者再次下载整个应用。 3、外观和使用感受与原生平台更加融为一体应用图标被放置在主屏幕上,应用可以全屏运行等。 凭借系统通知和推送消息与用户保持连接,对用户产生更多的吸引力,并且提高转换效率。 诚然,从零开始研发PWA应用会有一定的成本,但如果我们本身就拥有基于Web的站点,那么就可以通过增加对应的配置文件和服务进行升级操作,直接拥有PWA应用。HTTPS服务 首先PWA要求站点的请求方式为HTTPS,如果是生产环境,可以通过为Nginx服务器配置SSL的方式进行适配,但是线下环境测试PWA时就有点费劲了,所以通过openssl工具为本地域名localhost做自签证书:opensslreqx509outlocalhost。crtkeyoutlocalhost。keynewkeyrsa:2048nodessha256days3650subjCNlocalhostextensionsEXTconfig(printf〔dn〕CNlocalhost〔req〕distinguishednamedn〔EXT〕subjectAltNameDNS:localhostkeyUsagedigitalSignatureextendedKeyUsageserverAuth) 产出:localhost。crt和localhost。key文件,key是私用密钥openssl格式,通常是rsa算法。csr是证书请求文件,用于申请证书,在制作csr文件的时,必须使用自己的私钥来签署申,还可以设定一个密钥。 将文件放到项目的根目录下,随后在构建项目服务的时候配置即可,以Tornado为例:serverhttpserver。HTTPServer(app,xheadersTrue,ssloptions{certfile:。localhost。crt,keyfile:。localhost。key,})指定端口server。listen(443) 这里通过设置ssloptions参数来导入私钥和证书,同时将端口改为HTTPS默认端口号443。如此,在本地也可以对PWA进行测试了,当然了,如果不需要本地操作,也可以跳过这步。manifest。json配置文件 为了实现PWA应用添加至桌面的功能,除了要求站点支持HTTPS之外,还需要准备manifest。json文件去配置应用的图标、名称等信息。 以本站为例,在站点根目录创建manifest。json文件:{name:刘悦的技术博客,shortname:刘悦的技术博客,description:刘悦的技术博客,icons:〔{src:https:v3u。cnv3uPublicimagespwa192。png,sizes:192x192,type:imagepng},{src:https:v3u。cnv3uPublicimagespwa512。png,sizes:512x512,type:imagepng}〕,backgroundcolor:FFF,themecolor:FFF,display:standalone,orientation:portrait,starturl:,scope:} 由上至下,依次是PWA应用的名称、描述、图标文件、banner颜色、显示方式、开始页面的链接和PWA的作用域。为此我们需要提供两张不同分辨率的站点图标文件: ServiceWorker服务 ServiceWorker是一个注册在指定源和路径下的事件驱动型WebWorker。它充当了Web应用程序与浏览器之间的代理服务器,进行资源在文件级别下的缓存与操控,拦截页面请求,实现在不同的情况下对不同请求的响应策略。 ServiceWorker本质上就是一个WebWorker,因此它具有WebWorker的特点:无法操作DOM、脱离主线程、独立上下文。 ServiceWorker还具有这些特点:只能在Https下使用、运行在浏览器后台,不受页面刷新影响、更强大的离线缓存能力(使用CacheAPI)、请求拦截能力、完全异步,不能使用同步API、持续运行,第一次访问页面后,ServiceWorker就会安装激活并持续运行,直到手动销毁。 以本站为例,在站点根目录创建sw。js文件,注意ServiceWorker文件位置一定得在根目录,如果不在根目录也要通过重写或者url映射让其可以通过根目录路径进行访问,如:https:v3u。cnsw。js,否则浏览器会检测不到ServiceWorker服务:varCACHENAMEv3ucachev1;varurlsToCache〔,v3uPubliccsstidymin。css〕;self。addEventListener(install,function(event){event。waitUntil(caches。open(CACHENAME)。then(function(cache){console。log(Opencache);returncache。addAll(urlsToCache);})。then(function(){self。skipWaiting();}));}); 当我们为页面注册ServiceWorker后,ServiceWorker开始进行安装,安装成功之后,会在worker中触发install事件;如果安装失败,则进入废弃状态。 如果ServiceWorker逻辑文件更新(相关资源文件变动或者内部逻辑更新等),ServiceWorker会重新安装,如果这个时候,页面依然存在激活状态下的worker(旧的ServiceWorker),那么新的worker会进入waiting状态进行等待,直到我们主动去操作worker强制其更新,或者等待用户关闭所有页面,这个时候新的worker才会进入到激活状态。 在install事件中,我们使用caches。open方法打开cache对象,并通过cache。addAll缓存所有我们列出的文件。如果ServiceWorker存在更新,我们使用skipWaiting跳过等待,直接强制新的worker进入激活状态。 随后,添加fetch事件:self。addEventListener(fetch,function(event){if(event。request。method!GET)return;event。respondWith(caches。match(event。request)。then(function(response){if(response){console。log(returncaches);returnresponse;}else{returnfetch(event。request)。catch(function(){if(。html。test(event。request。url))returncaches。match(htmlneterror。html);});}}))}); 这里只监听了全站的GET请求方式,即我们只希望控制资源请求。通过caches。match检查请求是否命中了缓存,如果命中,则直接返回缓存给用户,防止重复请求,节约资源。如果没有命中,则将使用fetch方法请求网络资源并返回给用户。当网络状态异常时(fetch()。catch()),返回404页面的缓存给用户,告知用户当前处于无网络状态,不能访问相关页面。指定了一些页面和文件进行缓存,我们希望用户在无网络的情况下只能访问到我们指定缓存的页面。 当然,还有另外一种情况,我们指定了一些页面进行缓存(常用页面),当用户访问到一些不常用页面时,再对其进行缓存。这样,我们可以对资源配置进行优化,不过多的占用用户本地资源去缓存所有页面,因为PWA的缓冲本身是存储到客户端的,对于非所有用户的常用页面,按需缓存:self。addEventListener(fetch,function(event){if(event。request。method!GET)return;event。respondWith(caches。match(event。request)。then(function(response){if(response){console。log(returncaches);returnresponse;}else{returnfetch(event。request)。then(function(res){varresponseToCacheres。clone();caches。open(CACHENAME)。then(function(cache){catch。put(event。request,responseToCache);})returnres;});}}))}); 至此,ServiceWorker服务文件就撰写完成了。生产环境上线配置: 分别将manifest。json和sw。js文件分别上传到生产环境之后,在页面的head标签中进行声明:linkrelmanifesthrefmanifest。json 声明后,注意访问一下是否正确返回:https:v3u。cnmanifest。json 随后在页面中注册ServiceWorker服务: 这里首先判断当前浏览器的navigator是否支持serviceWorker,随后使用navigator。serviceWorker。register函数来注册ServiceWorker。其中,参数为要执行的worker逻辑文件路径,注意这个路径是基于origin的,而非当前文件。 接着键入组合键,打开chrome浏览器的开发者工具: Mac系统上的I Win系统上的F12CtrlShiftI 在Chrome的应用标签下进行检查,看应用清单有没有读出你的PWA应用信息配置文件: 随后在serviceWorker标签下检查serviceWorker是否正确运行: 接着访问站点,在地址栏即可添加PWA应用: 访问效果: 结语 渐进式增强和响应式设计已经可以让我们构建对移动端非常友好的站点,而PWA则又在我们的身后轻轻地推了一把,黄河之水源可滥觞,星星之火正在燎原,一年以内,我们都将感到PWA的灼人温度。