找回密码
 立即注册
首页 业界区 安全 SvelteKit 最新中文文档教程(16)—— Service workers ...

SvelteKit 最新中文文档教程(16)—— Service workers

颛孙中 7 天前
前言

Svelte,一个语法简洁、入门容易,面向未来的前端框架。
从 Svelte 诞生之初,就备受开发者的喜爱,根据统计,从 2019 年到 2024 年,连续 6 年一直是开发者最感兴趣的前端框架 No.1
1.png

Svelte 以其独特的编译时优化机制著称,具有轻量级高性能易上手等特性,非常适合构建轻量级 Web 项目
为了帮助大家学习 Svelte,我同时搭建了 Svelte 最新的中文文档站点。
如果需要进阶学习,也可以入手我的小册《Svelte 开发指南》,语法篇、实战篇、原理篇三大篇章带你系统掌握 Svelte!
欢迎围观我的“网页版朋友圈”、加入“冴羽·成长陪伴社群”,踏上“前端大佬成长之路”。
Service workers

Service workers 作为代理服务端,处理应用程序内部的网络请求。这使得您的应用程序能够离线工作,但即使您不需要离线支持(或由于您正在构建的应用程序类型而无法实现它),使用 service workers 预缓存构建的 JS 和 CSS 来加快导航速度,通常也是值得的。
在 SvelteKit 中,如果您有一个 src/service-worker.js 文件(或 src/service-worker/index.js),它将被打包并自动注册。如果需要,您可以更改 service worker 的位置。
如果您需要使用自己的逻辑注册 service worker 或使用其他解决方案,可以禁用自动注册。默认注册看起来类似这样:
  1. if ('serviceWorker' in navigator) {
  2.         addEventListener('load', function () {
  3.                 navigator.serviceWorker.register('./path/to/service-worker.js');
  4.         });
  5. }
复制代码
Service Worker 内部

在 service worker 内部,您可以访问 $service-worker 模块,它为您提供所有静态资源、构建文件和预渲染页面的路径。您还会获得一个应用程序版本字符串,可用于创建唯一的缓存名称,以及部署的 base 路径。如果您的 Vite 配置指定了 define(用于全局变量替换),这也将应用于 service workers 以及服务端/客户端构建。
以下示例会尽可能早的缓存构建的应用程序和 static 中的所有文件,并在访问时缓存所有其他请求。这将使每个页面在访问后都能离线工作。
  1. // @errors: 2339
  2. /// <reference types="@sveltejs/kit" />
  3. import { build, files, version } from '$service-worker';
  4. // 为此部署创建唯一的缓存名称
  5. const CACHE = `cache-${version}`;
  6. const ASSETS = [
  7.         ...build, // 应用程序本身
  8.         ...files // `static` 中的所有内容
  9. ];
  10. self.addEventListener('install', (event) => {
  11.         // 创建新缓存并添加所有文件
  12.         async function addFilesToCache() {
  13.                 const cache = await caches.open(CACHE);
  14.                 await cache.addAll(ASSETS);
  15.         }
  16.         event.waitUntil(addFilesToCache());
  17. });
  18. self.addEventListener('activate', (event) => {
  19.         // 从磁盘删除以前的缓存数据
  20.         async function deleteOldCaches() {
  21.                 for (const key of await caches.keys()) {
  22.                         if (key !== CACHE) await caches.delete(key);
  23.                 }
  24.         }
  25.         event.waitUntil(deleteOldCaches());
  26. });
  27. self.addEventListener('fetch', (event) => {
  28.         // 忽略 POST 请求等
  29.         if (event.request.method !== 'GET') return;
  30.         async function respond() {
  31.                 const url = new URL(event.request.url);
  32.                 const cache = await caches.open(CACHE);
  33.                 // `build`/`files` 始终可以从缓存中提供服务
  34.                 if (ASSETS.includes(url.pathname)) {
  35.                         const response = await cache.match(url.pathname);
  36.                         if (response) {
  37.                                 return response;
  38.                         }
  39.                 }
  40.                 // 对于其他所有内容,首先尝试网络
  41.                 // 但如果我们离线,则回退到缓存
  42.                 try {
  43.                         const response = await fetch(event.request);
  44.                         // 如果我们离线,fetch 可能返回非 Response 值
  45.                         // 而不是抛出错误 - 我们不能将这个非 Response 传递给 respondWith
  46.                         if (!(response instanceof Response)) {
  47.                                 throw new Error('invalid response from fetch');
  48.                         }
  49.                         if (response.status === 200) {
  50.                                 cache.put(event.request, response.clone());
  51.                         }
  52.                         return response;
  53.                 } catch (err) {
  54.                         const response = await cache.match(event.request);
  55.                         if (response) {
  56.                                 return response;
  57.                         }
  58.                         // 如果没有缓存,就直接报错
  59.                         // 因为我们无法对这个请求做任何响应
  60.                         throw err;
  61.                 }
  62.         }
  63.         event.respondWith(respond());
  64. });
复制代码
[!NOTE] 缓存时要小心!在某些情况下,过时的数据可能比离线时无法获取的数据更糟糕。由于浏览器会在缓存太满时清空缓存,因此您还应该谨慎缓存大型资源,如视频文件。
在开发过程中

service worker 在生产环境中会被打包,但在开发过程中不会。因此,只有支持 service workers 中的模块 的浏览器才能在开发时使用它们。如果您手动注册 service worker,在开发时需要传递 { type: 'module' } 选项:
  1. import { dev } from '$app/environment';
  2. navigator.serviceWorker.register('/service-worker.js', {
  3.         type: dev ? 'module' : 'classic'
  4. });
复制代码
[!NOTE] 在开发环境中,build 和 prerendered 是空数组
类型安全

为 service workers 设置适当的类型需要一些手动设置。在您的 service-worker.js 中,在文件顶部添加以下内容:
  1. /// <reference types="@sveltejs/kit" />
  2. /// <reference no-default-lib="true"/>
  3. /// <reference lib="esnext" />
  4. /// <reference lib="webworker" />
  5. const sw = /** @type {ServiceWorkerGlobalScope} */ (/** @type {unknown} */ (self));
复制代码
  1. /// <reference types="@sveltejs/kit" />
  2. /// <reference no-default-lib="true"/>
  3. /// <reference lib="esnext" />
  4. /// <reference lib="webworker" />
  5. const sw = self as unknown as ServiceWorkerGlobalScope;
复制代码
这会禁用对 service worker 中不可用的 DOM 类型(如 HTMLElement)的访问,并实例化正确的全局变量。将 self 重新赋值给 sw 允许您在此过程中进行类型转换(有几种方法可以做到这一点,但这是最简单的,不需要额外的文件)。在文件的其余部分使用 sw 而不是 self。
对 SvelteKit 类型的引用确保 $service-worker 导入具有适当的类型定义。如果您导入 $env/static/public,您要么必须使用 // @ts-ignore 注释导入,要么添加 ///  到引用类型中。
其他解决方案

SvelteKit 的 service worker 实现故意保持低级别。如果您需要更全功能但也更有主见的解决方案,我们建议查看像 Vite PWA 插件 这样的解决方案,它使用 Workbox。有关 service workers 的更多一般信息,我们推荐 MDN web 文档。
Svelte 中文文档

点击查看中文文档:SvelteKit Service workers
系统学习 Svelte,欢迎入手小册《Svelte 开发指南》。语法篇、实战篇、原理篇三大篇章带你系统掌握 Svelte!
此外我还写过 JavaScript 系列、TypeScript 系列、React 系列、Next.js 系列、冴羽答读者问等 14 个系列文章, 全系列文章目录:https://github.com/mqyqingfeng/Blog
欢迎围观我的“网页版朋友圈”、加入“冴羽·成长陪伴社群”,踏上“前端大佬成长之路”。

来源:程序园用户自行投稿发布,如果侵权,请联系站长删除
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
您需要登录后才可以回帖 登录 | 立即注册