使用传统前端技术增强客户端缓存能力

本文使用「署名 4.0 国际 (CC BY 4.0)」许可协议,欢迎转载、或重新修改使用,但需要注明来源。 署名 4.0 国际 (CC BY 4.0)

本文作者: 苏洋

创建时间: 2018年10月16日 统计字数: 3721字 阅读时间: 8分钟阅读 本文链接: https://soulteary.com/2018/10/16/use-traditional-front-end-technology-to-enhance-client-caching-capabilities.html


使用传统前端技术增强客户端缓存能力

前几天重构之后,Lighthouse 中有一个评分让我念念不忘: ProgressiveWebApp

PWA 不算一个新话题了,所以概念性的东西和 API 我就不多做介绍,下面简单介绍一个无干预更新的缓存方案,整体代码量在一百行以内,如果你也想在不“大动干戈”的情况下对站点或者 WebApp 进行性能提升的话,可以了解一下。

浏览器客户端代码

说到 PWA ,我们能直接想到的,无非是 增强缓存 和 推送能力。而这两个能力,都是 ServiceWorker API 实现的。(添加桌面图标这个需求,我不需要,就不介绍了,感兴趣可以自行搜索)

我在之前的重构文章中有简单聊过访客数据,其中有一部分访客使用的客户端并不支持 ServiceWorker,所以在使用它的时候,需要使用能力探测的方式引入,比如:

 
  1. try {
  2. ('serviceWorker' in navigator) && navigator.serviceWorker.register('/sw.js');
  3. } catch (error) {
  4. console.error(error);
  5. }

当然,构建压缩之后,你得到的结果应该是 drop console 的最简代码。不过,如果你不确定你的运行环境是否有问题,可以使用下面带有调试日志的版本。

 
  1. try {
  2. if ('serviceWorker' in navigator) {
  3. navigator.serviceWorker.register('/sw.min.js').then(function(registration) {
  4. console.log('ServiceWorker registration successful with scope: ', registration.scope);
  5. }).catch(function(err) {
  6. console.error('ServiceWorker registration failed: ', err);
  7. });
  8. }
  9. } catch (error) {
  10. console.error(error);
  11. }

ServiceWorker 客户端代码

上面介绍浏览器客户端代码的时候,有引入一个外部脚本依赖 sw.js

在分享代码之前,有做过 PWA 相关项目或者了解的同学,可能或多或少会在 增强缓存 这个地方被坑到,比如:缓存无法更新、缓存内容过多无法写入。

缓存无法更新有一个简单有效的解决方案:定时切换缓存使用的 Store。如果再引入当前时间这个因素,可以保障缓存使用的 Store 不存在资源争抢的问题。

结合站点内容特点,配合定期清理缓存的脚本,可以将缓存数量控制在一定的范围之内。

这里提供一个小思路,对服务端资源进行二次缓存的时候,可以设定一个最大缓存时间的策略,而这里有两个方案:

  • 对每个资源设定缓存 TTL
  • 对所有资源设定统一 TTL

我个人选择第二个方案,牺牲一定的缓存复用,但是有效降低资源之间的版本管理的复杂度。

而需要缓存的资源一般分为两类:

  • 短时间缓存:页面或者页面片段
  • 相对长时间缓存:图片等媒体资源,或者有一定跨页面通用能力的脚本和样式资源

这里以10分钟(调试模式单位替换为秒)为一个时间段,为短时间缓存的资源进行缓存。设定星期数为其他资源进行缓存周期。

这里我们不必过分处理 install 和 active 事件,只需要统一在 fetch 事件中进行缓存更新和清理即可,比如下面这样:

 
  1. function weekId() {
  2. var now = (new Date());
  3. var dt = new Date(now.getFullYear(), 0, 1);
  4. return Math.ceil((((now - dt) / 86400000) + dt.getDay() + 1) / 7);
  5. }
  6.  
  7. var isDevMode = false;
  8. var CACHE_LONG_TTL = 'WEEK_' + weekId();
  9. var CACHE_TINY_TTL = 'TINY_';
  10.  
  11. function cleanCache(whiteList) {
  12. return caches.keys().then(function(buckets) {
  13. return Promise.all(buckets.filter(function(bucket) {
  14. return whiteList.indexOf(bucket) === -1;
  15. }).map(function(bucket) {
  16. return caches.delete(bucket);
  17. }));
  18. });
  19. }
  20.  
  21. self.addEventListener('activate', function(event) {
  22. event.waitUntil(cleanCache([CACHE_LONG_TTL]));
  23. });
  24.  
  25. self.addEventListener('fetch', function(event) {
  26.  
  27. var url = new URL(event.request.url);
  28. if (url.protocol.toLowerCase() !== 'https:') return;
  29.  
  30. var now = new Date;
  31. var CACHE_INTERVAL = Math.floor((isDevMode ? now.getSeconds() : now.getMinutes()) / 10);
  32. var CACHE_KEY = CACHE_TINY_TTL + CACHE_INTERVAL;
  33.  
  34. if (url.pathname.endsWith('/') ||
  35. url.pathname.endsWith('.html') ||
  36. url.pathname.endsWith('.md') ||
  37. url.pathname.endsWith('/feed/') ||
  38. url.pathname.endsWith('/feed') ||
  39. url.pathname.endsWith('sw.js')) {
  40.  
  41. cleanCache([CACHE_LONG_TTL, CACHE_KEY]);
  42.  
  43. event.respondWith(caches.open(CACHE_KEY).then(function(cache) {
  44. return cache.match(url).then(function(response) {
  45. if (response) return response;
  46. return fetch(event.request).then(function(response) {
  47. cache.put(event.request, response.clone());
  48. return response;
  49. });
  50. });
  51. }));
  52.  
  53. } else {
  54. event.respondWith(caches.open(CACHE_LONG_TTL).then(function(cache) {
  55. return cache.match(event.request).then(function(response) {
  56. if (response) return response;
  57. return fetch(event.request).then(function(response) {
  58. cache.put(event.request, response.clone());
  59. return response;
  60. });
  61. });
  62. }));
  63. }
  64.  
  65. });

如果你的访客有不少是 Chromev50 和 FireFoxv46 以下的客户端,那么你可能还需要引入下面的 Polyfill : https://github.com/dominiccooney/cache-polyfill/blob/master/index.js

当然,在引入的时候,建议还是进行压缩,进一步提高页面性能(哪怕它是独立于客户端脚本另外的异步进程)。

当一切都完成之后,如果顺利的话,你的网站的将支持有限时间(本文是10min)的离线访问,以及重复访问时更好的响应能力。

另外,你也不需要担心 sw.js 被缓存后,这个站点变成“完全离线”。因为 sw.js 按照规范可以保证它的最长存在生命周期是 1天,也就是说,未来你的策略更新最多延迟1天。

最后

再次跑分,发现 Lighthouse 已经全面绿色评价了,不过前文中我提过,我不太需要把网站变为纯粹的 PWA 应用,没有去设定 mainfest ,所以,依旧有 4 点改进建议。

  1. 用户不能够“安装” Web App。
  2. 没有定义启动屏幕(仿客户端体验)。
  3. 没有定义顶栏的主题色。
  4. viewport 没有优化。

这些后面再说吧。

— EOF

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值