serviceWorker生命周期、请求代理与通信'

本文主要讲 serviceWorker 生命周期和挂载、卸载等问题,适合对 serviceWorker 的作用有所了解但是具体细节不是特别清楚的读者

以下所有分析基于 Chrome V63

serviceWorker的挂载

先来一段代码感受serviceWorker注册:

if ('serviceWorker' in navigator) {
      window.addEventListener('load', function () {
          navigator.serviceWorker.register('/sw.js', {scope: '/'})
              .then(function (registration) {
                  // 注册成功
                  console.log('ServiceWorker registration successful with scope: ', registration.scope);
              })
              .catch(function (err) {
                  // 注册失败:(
                  console.log('ServiceWorker registration failed: ', err);
              });
      });
}

通过上述代码,我们定义在/sw.js里的内容就会生效(对于当前页面之前没有 serviceWorker 的情况而言,我们注册的 serviceWorker 肯定会生效,如果当前页面已经有了我们之前注册的 serviceWorker,这个时候涉及到 serviceWorker的更新机制,下文详述)

如果我们在sw.js没有变化的情况下刷新这个页面,每次还是会有注册成功的回调以及相应的log输出,但是这个时候浏览器发现我们的 serviceWorker 并没有发生变化,并不会重置一遍 serviceWorker

serviceWorker更新

我们如果想更新一个 serviceWorker,根据我们的一般web开发策略,可能会想到以下几种策略:

  • 仅变更文件名(比如把sw.js变成sw-v2.js或者加一个hash)
  • 仅变更文件内容(仅仅更新sw.js的内容,文件名不变)
  • 同时变更:同时执行以上两条

在这里,我可以很负责的告诉你,变更serviceWorker文件名绝对不是一个好的实践,浏览器判断 serviceWorker 是否相同基本和文件名没有关系,甚至有可能还会造成浏览器抛出404异常(因为找不到原来的文件名对应的文件了)。

所以我们只需要变更内容即可,实际上,我们每次打开或者刷新该页面,浏览器都会重新请求一遍 serviceWorker 的定义文件,如果发现文件内容和之前的不同了,这个时候:

(下文中,我们使用“有关 tab”来表示受 serviceWorker 控制的页面,刷新均指普通刷新(F5/CommandR)并不指Hard Reload)

  • 这个新的 serviceWorker 就会进入到一个 “waiting to activate” 的状态,并且只要我们不关闭这个网站的所有tab(更准确地说,是这个 serviceWorker 控制的所有页面),新的 serviceWorker 始终不会进入替换原有的进入到 running 状态(就算我们只打开了一个有关 tab,直接刷新也不会让新的替换旧的)。

  • 如果我们多次更新了 serviceWorker 并且没有关闭当前的 tab 页面,那么新的 serviceWorker 就会挤掉原先处于第二顺位(waiting to activate)的serviceWorker,变成waiting to activate状态

也就是说,我们只有关闭当前旧的 serviceWorker 控制的所有页面 的所有tab,之后浏览器才会把旧的 serviveWorker 移除掉,换成新的,再打开相应的页面就会使用新的了。

当然,也有一个特殊情况:如果我们在新的 serviceWorker 使用了self.skipWaiting();,像这样:

self.addEventListener('install', function(event) {
    self.skipWaiting();
});

这个时候,要分为以下两种情况:

  • 如果当前我们只打开了一个有关 tab,这个时候,我们直接刷新,发现新的已经替换掉旧的了。
  • 如果我们当前打开了若干有关 tab,这个时候,无论我们刷新多少次,新的也不会替换掉旧的,只有我们一个一个关掉tab(或者跳转走)只剩下最后一个了,这个时候刷新,会让新的替换旧的(也就是上一种情况)

Chrome 的这种机制,防止了同一个页面先后被新旧两个不同的 serviceWorker 接管的情况出现。

手动更新

虽然说,在页面每次进入的时候浏览器都会检查一遍 serviceWorker 是否更新,但如果我们想要手动更新 serviceWorker 也没有问题:

navigator.serviceWorker.register("/sw.js").then(reg => {
  reg.update();
  // 或者 一段时间之后更新
});

这个时候如果 serviceWorker 变化了,那么会重新触发 install 执行一遍 install 的回调函数,如果没有变,就不会触发这个生命周期。

install 生命周期钩子

我们一般会在 sw.js 中,添加install的回调,一般在回调中,我们会进行缓存处理操作,像这样:

self.addEventListener('install', function(event) {
    console.log('[sw2] serviceWorker Installed successfully', event)

    event.waitUntil(
        caches.open('mysite-static-v1').then(function(cache) {
            return cache.addAll([
                '/stylesheets/style.css',
                '/javascripts/common.39c462651d449a73b5bb.js',
            ]);
        })
    )
}    

如果我们新打开一个页面,如果之前有 serviceWorker,那么会触发install,如果之前没有, 那么在 serviceWorker 装载后会触发 install

如果我们刷新页面,serviceWorker 和之前没有变化或者 serviceWorker 已经处在 waiting to activate,不会触发install,如果有变化,会触发install,但不会接管页面(上文中提到)。

activate 生命周期钩子

activate 在什么时候被触发呢?

如果当前页面没有 serviceworker ,那么会在 install 之后触发。

如果当前页面有 serviceWorker,并且有 serviceWorker更新,新的 serviceWorker 只会触发 install ,不会触发 activate

换句话说,当前变成 active 的 serviceWorker 才会被触发这个生命周期钩子

serviceWorker 代理请求

serviceWorker 代理请求相对来说比较好理解,以下是一个很简单的例子:

self.addEventListener('install', function(event) {
    console.log('[sw2] serviceWorker Installed successfully', event)

    event.waitUntil(
        caches.open('mysite-static-v1').then(function(cache) {
            return cache.addAll([
                '/stylesheets/style.css',
                '/javascripts/common.39c462651d449a73b5bb.js',
            ]);
        })
    );
});

self.addEventListener('fetch', function(event) {
    console.log('Handling fetch event for', event.request.url);
    // console.log('[sw2]fetch but do nothing')

    event.respondWith(
        // caches.match() will look for a cache entry in all of the caches available to the service worker.
        // It's an alternative to first opening a specific named cache and then matching on that.
        caches.match(event.request).then(function(response) {
            if (response) {
                console.log('Found response in cache:', response);

                return response;
            }

            console.log('No response found in cache. About to fetch from network...');

            // event.request will always have the proper mode set ('cors, 'no-cors', etc.) so we don't
            // have to hardcode 'no-cors' like we do when fetch()ing in the install handler.
            return fetch(event.request).then(function(response) {
                console.log('Response from network is:', response);

                return response;
            }).catch(function(error) {
                // This catch() will handle exceptions thrown from the fetch() operation.
                // Note that a HTTP error response (e.g. 404) will NOT trigger an exception.
                // It will return a normal response object that has the appropriate error code set.
                console.error('Fetching failed:', error);

                throw error;
            });
        })
    );
});

有两点要注意的:

我们如果这样代理了,哪怕没有 cache 命中,实际上也会在控制台写from serviceWorker,而那些真正由serviceWorker发出的请求也会显示,有一个齿轮图标,如下图:

第二点就是我们如果在 fetch 的 listener 里面 do nothing, 也不会导致这个请求直接假死掉的。

另外,通过上面的代码我们发现,实际上由于现在我们习惯给我们的文件资源加上 hash,所以我们基本上不可能手动输入需要缓存的文件列表,现在大多数情况下,我们都是借助 webpack 插件,完成这部分工作。

serviceWorker 和 页面之间的通信

serviceWorker向页面发消息:

sw.js:

self.clients.matchAll().then(clients => {
    clients.forEach(client => {
        console.log('%c [sw message]', 'color:#00aa00', client)
        client.postMessage("This message is from serviceWorker")
    })
})

主页面:

navigator.serviceWorker.addEventListener('message', function (event) {
    console.log('[Main] receive from serviceWorker:', event.data, event)
});

当然,这里面是有坑的:

  • 主界面的事件监听需要等serviceWorker注册完毕后,所以一般navigator.serviceWorker.register的回调到来之后再进行注册(或者延迟足够的时间)。
  • 如果在主界面事件监听还没有注册成功的时候 serviceWorker 发送消息,自然是收不到的。如果我们把 serviceWorker 直接写在 install 的回调中,也是不能被正常收到的。

从页面向 serviceWorker 发送消息:

主页面:

navigator.serviceWorker.controller && navigator.serviceWorker.controller.postMessage('hello serviceWorker');

sw.js:
self.addEventListener('message', function (event) {
    console.log("[sw from main]",event.data); // 输出:'sw.updatedone'
});

同样的,这也要求主界面的代码需要等到serviceWorker注册完毕后触发,另外还有一点值得注意, serviceWorker 的事件绑定代码要求主界面的serviceWorker已经注册完毕后才可以。

也就是说,如果当前页面没有该serviceWorker 第一次注册是不会收到主界面接收到的消息的。

记住,只有当前已经在 active 的 serviceWorker, 才能和主页面收发消息等。

以上就是和 serviceWorker 有关的一些内容,在下一篇文章中,我会对PWA 添加至主屏幕等功能进行总结