来用Service worker吧
admin
2024-03-27 17:02:48
0

之前说了笔者写了一个微前端框架。在微前端中子应用切换前要先获取到子应用的资源(比如js、css)和 html 片段进行加载。那么这里就涉及到一个“老生常谈”的话题:资源缓存

为了缓存子应用的 js、css资源,笔者分两步进行:首先是在主应用加载时“预加载”子应用:

export const prefetch = async () => {// 获取到所有子应用的列表,但不包括当前正在显示的const list = getList().filter(item => !window.location.pathname.startsWith(item.activeRule))// 预加载剩下的所有子应用await Promise.all(list.map(async item => await parseHtml(item.entry, item.name)))
}

然后是在资源请求前后判断缓存、存入缓存:

const cache = {} //根据子应用的name做缓存export const parseHtml = async (entry, name) => {if(cache[name]) {return cache[name]}// 资源加载其实是一个get请求,我们去模拟这个过程const html = await fetchResource(entry)let allScripts = []//...// 抽离标签、link、script(src、代码)const [dom, scriptUrl, script] = await getResources(div, entry)// 获取所有的js资源const fetchedScripts = await Promise.all(scriptUrl.map(async item => await fetchResource(item)))allScripts = script.concat(fetchedScripts)cache[name] = [dom, allScripts]//...
}

这样就达到了提高响应速度的效果。

那在普通项目中呢?
Service Worker API 能够很好的帮助我们完成这个功能:资源缓存、请求缓存。

拿静态资源来说,使用时先注册:

function register() {if('serviceWorker' in navigator) {// 注册,可以给一个scope参数作为作用域navigator.serviceWorker.register('/worker文件路径', {scope: '作用域路径'}).then((res) => {if(res.installing) {// 注册成功} else if (res.waiting) {// 注册过了} else if (res.active) {// 已经开启}}).catch((err) => {console.log('register failed with: ' + err)})}
}

然后进入到 worker 文件中。

一个service注册成功后必然进行install 事件,这里我们可以进行一些静态资源的 cache,以防止在后面的请求时依然对这些资源发送请求!

self.addEventListener('install', function(event) {event.waitUntil(caches.open("缓存名(key)").then(function(cache) {console.log('[SW]: Opened cache');return cache.addAll(静态资源列表);}));
});

这里还有一个需要注意的地方:
如果是第一次加载 sw ,在install后,会直接进入activated阶段,而如果 sw 进行更新,情况就会显得复杂一些:当新的 sw 进入install阶段,而老的那个还处于工作状态,新的那个就会进入waiting阶段。只有等到老的被terminated后,新的才能正常替换老的那个的工作。

如果你不想等待,可以试试把它加到install的“最前面”:

//跳过等待过程
self.skipWaiting();

然后就进入了activated阶段,激活 sw 工作。
activated阶段我们主要用于处理缓存,比如更新存储在cache中的key和value:

self.addEventListener('activate', function(event) {//清除cache中原来老的一批相同key的数据var setOfExpectedUrls = new Set(缓存列表的key-value二维数组.values());event.waitUntil(caches.open(缓存名).then(function(cache) {return cache.keys().then(function(existingRequests) {return Promise.all(existingRequests.map(function(existingRequest) {if (!setOfExpectedUrls.has(existingRequest.url)) {//cache中删除指定对象return cache.delete(existingRequest);}}));});}).then(function() {//self相当于webworker线程的当前作用域//当一个 service worker 被初始注册时,页面在下次加载之前不会使用它。claim() 方法会立即控制这些页面//从而更新客户端上的serviceworkerreturn self.clients.claim();}));
});

然后最关键的就是 fetch 阶段了。

var isPathWhitelisted = function(whitelist, absoluteUrlString) {// If the whitelist is empty, then consider all URLs to be whitelisted.if (whitelist.length === 0) {return true;}// Otherwise compare each path regex to the path of the URL passed in.var path = (new URL(absoluteUrlString)).pathname;return whitelist.some(function(whitelistedPathRegex) {return path.match(whitelistedPathRegex);});
};self.addEventListener('fetch', function(event) {if (event.request.method === 'GET') {// 标识位,用来判断是否需要缓存var shouldRespond;// 处理参数// 再次验证,判断其是否是一个navigation类型的请求var navigateFallback = '';if (!shouldRespond &&navigateFallback &&(event.request.mode === 'navigate') &&isPathWhitelisted([], event.request.url)) {url = new URL(navigateFallback, self.location).toString();shouldRespond = urlsToCacheKeys.has(url);}// 如果标识位为trueif (shouldRespond) {event.respondWith(caches.open(cacheName).then(function(cache) {//去缓存cache中找对应的url的值return cache.match(urlsToCacheKeys.get(url)).then(function(response) {//如果找到了,就返回valueif (response) {return response;}throw Error('The cached response that was expected is missing.');});}).catch(function(e) {// 如果没找到则请求该资源、return fetch(event.request);}));}}
});

先校验请求url和参数。然后判断 cache 中是否有缓存,没有的话就请求资源。

注意:上面说的是“静态资源缓存”,也就是缓存列表是固定的。但是如果你想要缓存普通请求或者其他资源,就不能只是简单的fetch一下了:

self.addEventListener('fetch', function (evt) {//处理// 在发起请求时候会触发fetch事件evt.respondWith(caches.match(evt.request).then(function (response) {// 如果 sw 已经保存了请求的响应,直接返回响应,减少http请求if (response !== undefined) {return response}// 不存在需要发起请求return fetch(evt.request).then((httpRes) => {if (!httpRes || httpRes !== 200) {// 请求出错则直接返回错误信息return httpRes}// 将响应复制一份const httpResClone = httpRes.clone()// 并且保存到安装时候的缓存对象里caches.open(缓存名).then((cache) => {cache.put(evt.request, httpResClone)})return httpRes})}))
})

当cache里面没有缓存,则使用fetch发起请求,这个Fetch发起请求的是用来代替XMLHttpRequest来发请求的方案;当请求响应了错误直接返回错误信息,当请求响应成功的情况,更新cache缓存,将新的响应存入cache缓存中,下次在访问就直接从缓存中读取。

相关内容

热门资讯

linux入门---制作进度条 了解缓冲区 我们首先来看看下面的操作: 我们首先创建了一个文件并在这个文件里面添加了...
C++ 机房预约系统(六):学... 8、 学生模块 8.1 学生子菜单、登录和注销 实现步骤: 在Student.cpp的...
A.机器学习入门算法(三):基... 机器学习算法(三):K近邻(k-nearest neigh...
数字温湿度传感器DHT11模块... 模块实例https://blog.csdn.net/qq_38393591/article/deta...
有限元三角形单元的等效节点力 文章目录前言一、重新复习一下有限元三角形单元的理论1、三角形单元的形函数(Nÿ...
Redis 所有支持的数据结构... Redis 是一种开源的基于键值对存储的 NoSQL 数据库,支持多种数据结构。以下是...
win下pytorch安装—c... 安装目录一、cuda安装1.1、cuda版本选择1.2、下载安装二、cudnn安装三、pytorch...
MySQL基础-多表查询 文章目录MySQL基础-多表查询一、案例及引入1、基础概念2、笛卡尔积的理解二、多表查询的分类1、等...
keil调试专题篇 调试的前提是需要连接调试器比如STLINK。 然后点击菜单或者快捷图标均可进入调试模式。 如果前面...
MATLAB | 全网最详细网... 一篇超超超长,超超超全面网络图绘制教程,本篇基本能讲清楚所有绘制要点&#...
IHome主页 - 让你的浏览... 随着互联网的发展,人们越来越离不开浏览器了。每天上班、学习、娱乐,浏览器...
TCP 协议 一、TCP 协议概念 TCP即传输控制协议(Transmission Control ...
营业执照的经营范围有哪些 营业执照的经营范围有哪些 经营范围是指企业可以从事的生产经营与服务项目,是进行公司注册...
C++ 可变体(variant... 一、可变体(variant) 基础用法 Union的问题: 无法知道当前使用的类型是什...
血压计语音芯片,电子医疗设备声... 语音电子血压计是带有语音提示功能的电子血压计,测量前至测量结果全程语音播报࿰...
MySQL OCP888题解0... 文章目录1、原题1.1、英文原题1.2、答案2、题目解析2.1、题干解析2.2、选项解析3、知识点3...
【2023-Pytorch-检... (肆十二想说的一些话)Yolo这个系列我们已经更新了大概一年的时间,现在基本的流程也走走通了,包含数...
实战项目:保险行业用户分类 这里写目录标题1、项目介绍1.1 行业背景1.2 数据介绍2、代码实现导入数据探索数据处理列标签名异...
记录--我在前端干工地(thr... 这里给大家分享我在网上总结出来的一些知识,希望对大家有所帮助 前段时间接触了Th...
43 openEuler搭建A... 文章目录43 openEuler搭建Apache服务器-配置文件说明和管理模块43.1 配置文件说明...