2019 前端性能优化年度总结 — 第四部分
栏目: JavaScript · 发布时间: 5年前
内容简介:让 2019 来得更迅速吧!你现在阅读的是 2019 年前端性能优化年度总结,始于 2016。要了解你首先要处理什么。列出你全部的静态资源清单(JavaScript、图片、字体、第三方脚本以及页面上的大模块:如轮播图、复杂的信息图表和多媒体内容),并将它们分组。新建一个电子表格。定义旧版浏览器的基本
让 2019 来得更迅速吧!你现在阅读的是 2019 年前端性能优化年度总结,始于 2016。
- [译] 2019 前端性能优化年度总结 — 第一部分
- [译] 2019 前端性能优化年度总结 — 第二部分
- [译] 2019 前端性能优化年度总结 — 第三部分
- [译] 2019 前端性能优化年度总结 — 第四部分
- [译] 2019 前端性能优化年度总结 — 第五部分
- [译] 2019 前端性能优化年度总结 — 第六部分
目录
-
- 23.重温优秀的“符合最低要求”技术
- 24.解析 JavaScript 是耗时的,所以让它体积小
- 25.使用了摇树、作用域提升和代码分割吗
- 26.可以将 JavaScript 切换到 Web Worker 中吗?
- 27.可以将 JavaScript 切换到 WebAssembly 中吗?
- 28.是否使用了 AOT 编译?
- 29.仅将遗留代码提供给旧版浏览器
- 30.是否使用了 JavaScript 差异化服务?
- 31.通过增量解耦识别和重写遗留代码
- 32.识别并删除未使用的 CSS/JS
- 33.减小 JavaScript 包的大小
- 34.是否使用了 JavaScript 代码块的预测预获取?
- 35.从针对你的目标 JavaScript 引擎进行优化中获得好处
- 36.使用客户端渲染还是服务器端渲染?
- 37.约束第三方脚本的影响
- 38.设置 HTTP 缓存标头
构建优化
22. 确定优先级
要了解你首先要处理什么。列出你全部的静态资源清单(JavaScript、图片、字体、第三方脚本以及页面上的大模块:如轮播图、复杂的信息图表和多媒体内容),并将它们分组。
新建一个电子表格。定义旧版浏览器的基本 核心 体验(即完全可访问的核心内容)、现代浏览器的 增强 体验(即更加丰富的完整体验)以及 额外功能 (可以延迟加载的非必需的资源:例如网页字体、不必要的样式、轮播脚本、视频播放器、社交媒体按钮和大图片)。不久前,我们发表了一篇关于“ 提升 Smashing 杂志网站性能 ”的文章,文中详细描述了这种方法。
在优化性能时,我们需要确定我们的优先事项。立即加载 核心体验 ,然后加载 增强体验 ,最后加载 额外功能 。
23. 重温优秀的“符合最低要求”技术
如今,我们仍然可以使用 符合最低要求(cutting-the-mustard)技术 将核心体验发送到旧版浏览器,并为现代浏览器提供增强体验。(译者注:关于 cutting-the-mustard 出处可以参考这篇文章。)该技术的一个更新版本将使用 ES2015 + 语法 <script type="module">
。现代浏览器会将脚本解释为 JavaScript 模块并按预期运行它,而旧版浏览器无法识别该属性并忽略它,因为它是未知的 HTML 语法。
现在我们需要谨记的是,单独的功能检测不足以做出该发送哪些资源到该浏览器的明智决定。就其本身而言, 符合最低要求 从浏览器版本中推断出设备的能力,今天已经不再有效了。
例如,发展中国家的廉价 Android 手机主要使用 Chrome 浏览器,尽管设备的内存和 CPU 功能有限,但其仍然达到了使用符合最低要求技术的标准。最终,使用 设备内存客户端提示报头 ,我们将能够更可靠地定位低端设备。在本文写作时,仅在 Blink 中支持该报头(通常用于客户端提示)。由于设备内存还有 一个已在 Chrome 中提供 的 JavaScript API,因此基于该 API 进行功能检测是一个选择,并且只有在不支持时才会再来使用符合最低要求技术( 感谢 Yoav! )。
24. 解析 JavaScript 是耗时的,所以让它体积小
在处理单页面应用程序时,我们需要一些时间来初始化应用程序,然后才能渲染页面。你的设置需要你的自定义解决方案,但可以留意能够加快首次渲染的模块和技术。例如,如何调试 React 性能、 消除常见的 React 性能问题 ,以及 如何提高 Angular 的性能 。通常,大多数性能问题都来自启动应用程序的初始解析时间。
JavaScript 有一个解析的成本 ,但很少仅是由于文件大小一个因素影响性能。解析和执行时间根据设备的硬件的不同有很大差异。在普通电话(Moto G4)上,1MB(未压缩)JavaScript 的解析时间约为 1.3-1.4s,移动设备上有 15-20% 的时间用于解析。在游戏中编译,仅仅在准备 JavaScript 就平均耗时 4 秒,在移动设备上首次有效绘制(First Meaningful Paint )之前大约需要 11 秒。原因:在低端移动设备上, 解析和执行时间很容易高出 2-5 倍 。
为了保证高性能,作为开发人员,我们需要找到编写和部署更少量 JavaScript 的方法。这就是为什么要详细检查每一个 JavaScript 依赖关系的原因。
有许多 工具 可以帮助你做出有关依赖关系和可行替代方案影响的明智决策:
- webpack-bundle-analyzer
- Source Map Explorer
- Bundle Buddy
- Bundlephobia
- Webpack size-plugin
- Import Cost for Visual Code
有一种有趣方法可以用来避免解析成本,它使用了 Ember 在 2017 年推出的二进制模板。使用该模板,Ember 用 JSON 解析代替 JavaScript 解析,这可能更快。( 感谢 Leonardo,Yoav! )
衡量 JavaScript 解析和编译时间 。我们可以使用综合测试工具和浏览器跟踪来跟踪解析时间,浏览器实现者正在谈论 将来把基于 RUM 的处理时间暴露出来 。也可以考虑使用 Etsy 的 DeviceTiming ,这是一个小工具,它允许你使用 JavaScript 在任何设备或浏览器上测量解析和执行时间。
底线:虽然脚本的大小很重要,但它并不是一切。随着脚本大小的增长,解析和编译时间不一定会线性增加。
25. 使用了摇树、作用域提升和代码分割吗
摇树(tree-shaking)是一种在webpack 中清理构建过程的方法,它仅将实际生产环境使用的代码打包,并排除没有使用的导入模块。使用 webpack 和 rollup,还可以使用作用域提升(scope hoisting),作用域提升使得 webpack 和 rollup 可以检测 import
链可以展开的位置,并将其转换为一个内联函数,并且不会影响代码。使用 webpack,我们也可以使用JSON Tree Shaking。
此外,你可能需要考虑学习如何 编写高效的 CSS 选择器 ,以及如何避免臃肿且耗时的样式。如果你希望更进一步,你还可以使用 webpack 来缩短 class 名,并使用作用域隔离在编译时 动态重命名 CSS class 名 。
代码拆分(code-splitting) 是另一个 webpack 功能,它将你的代码库拆分为按需加载的“块”。并非所有的 JavaScript 都必须立即下载、解析和编译。在代码中定义分割点后,webpack 可以处理依赖项和输出文件。它能够保持较小体积的初始下载,并在应用程序请求时按需请求代码。Alexander Kondrov 有一个 使用 webpack 和 React 应用代码分割的精彩介绍 。
考虑使用 preload-webpack-plugin ,它接受代码拆分的路由,然后提示浏览器使用 <link rel="preload">
或 <link rel="prefetch">
预加载它们。Webpack 内联指令还可以控制 preload
/ prefetch
。
在哪里定义分割点呢?通过跟踪代码查看使用了哪些 CSS/JavaScript 包,没有使用哪些包。Umar Hansa解释了如何使用 Devtools 的代码覆盖率工具来实现它。
如果你没有使用 webpack,请注意rollup 显示的结果明显优于 Browserify 导出。虽然我们参与其中,但你可能需要查看 rollup-plugin-closure-compiler 和 rollupify ,它将 ECMAScript 2015 模块转换为一个大型 CommonJS 模块 —— 因为根据你的包和模块系统的选择,小模块可能会有惊人高的成本。
26. 可以将 JavaScript 切换到 Web Worker 中吗?
为了减少对首次可交互时间(Time-to-Interactive)的负面影响,考虑将高耗时的 JavaScript 放到Web Worker 或通过 Service Worker 来缓存。
随着代码库的不断增长,UI 性能瓶颈将会出现,进而会降低用户的体验。主要 原因是 DOM 操作与主线程上的 JavaScript 一起运行 。通过web worker,我们可以将这些高耗时的操作移动到后台进程的另一线程上。Web worker 的典型用例是 预获取数据和渐进式 Web 应用程序 ,提前加载和存储一些数据,以便你在之后需要时使用它。而且你可以使用 Comlink 简化主页面和 worker 之间的通信。仍然还有一些工作要做,但我们已经做了很多了。
Workerize 让你能够将模块移动到 Web Worker 中,自动将导出的函数映射为异步代理。如果你正在使用 webpack,你可以使用 workerize-loader 。或者,也可以试试 worker-plugin 。
请注意,Web Worker 无权访问 DOM,因为 DOM 不是“线程安全的”,而且它们执行的代码需要包含在单独的文件中。
27. 可以将 JavaScript 切换到 WebAssembly 中吗?
我们还可以将 JavaScript 转换为WebAssembly,这是一种二进制指令格式,可以使用 C/C++/Rust 等高级语言进行编译。它的浏览器支持非常出色,最近它变得可行了,因为 JavaSript 和 WASM 之间的函数调用速度变得越来越快 ,至少在 Firefox 中是这样。
在实际场景中, JavaScript 似乎在较小的数组大小上比 WebAssembly 表现更好 ,而 WebAssembly 在更大的数组大小上比 JavaScript 表现更好。对于大多数 Web 应用程序,JavaScript 更适合,而 WebAssembly 最适合用于计算密集型 Web 应用程序,例如 Web 游戏。但是,如果切换到 WebAssembly 能否获得显着的性能改进,则可能值得研究。
如果你想了解有关 WebAssembly 的更多信息:
-
Lin Clark 为 WebAssembly 撰写了一个全面的系列文章,Milica Mihajlija概述了如何在浏览器中运行原生代码、为什么要这样做、以及它对 JavaScript 和 Web 开发的未来意味着什么。
-
Google Codelabs 提供了一份WebAssembly 简介,这是一个 60 分钟的课程,你将学习如何使用原生代码 —— 使用 C 并将其编译为 WebAssembly,然后直接在 JavaScript 调用它。
-
Alex Danilo 在他的 Google I/O 2017 演讲中 解释了 WebAssembly 及其工作原理 。此外,Benedek Gagyi 分享了一个关于 WebAssembly 的实际案例研究 ,特别是团队如何将其用作 iOS、Android 和网站的 C++ 代码库的输出格式。
Milica Mihajlija 提供了 WebAssembly 的工作原理及其有用的原因 的概述。 (预览大图)
28. 是否使用了 AOT 编译?
使用 AOT(ahead-of-time)编译器 将一些客户端渲染放到服务器,从而快速输出可用结果。最后,考虑使用 Optimize.js 来加速初始化加载时间,它包装了需要立即调用的函数(尽管现在这可能不是必需的了)。
来自 默认快速:现代加载最佳实践 ,作者是独一无二的 Addy Osmani。幻灯片第 76 页。
29. 仅将遗留代码提供给旧版浏览器
由于 ES2015 在现代浏览器中得到了非常好的支持 ,我们可以 使用 babel-preset-env
,仅转义尚未被我们的目标浏览器支持的那些 ES2015 + 特性。然后 设置两个构建 ,一个在 ES6 中,一个在 ES5 中。如上所述,现在所有主流浏览器都支持 JavaScript 模块,因此使用 script type =“module”
让支持 ES 模块的浏览器加载支持 ES6 的文件,而旧浏览器可以使用 script nomodule
加载支持 ES5 的文件。我们可以使用 Webpack ESNext Boilerplate 自动完成整个过程。
请注意,现在我们可以编写基于模块的 JavaScript,它可以原生地在浏览器里运行,无需编译器或打包工具。 <link rel="modulepreload">
header 提供了一种提前(和高优先级)加载模块脚本的方法。基本上,它能够很好地最大化使用带宽,通过告诉浏览器它需要获取什么,以便在这些长的往返期间不会卡顿。此外,Jake Archibald 发布了一篇详细的文章,其中包含了 需要牢记的 ES 模块相关内容 ,值得一读。
对于 lodash, 使用 babel-plugin-lodash
,通过它可以只加载你在源代码中使用的模块。你的其他依赖也可能依赖于其他版本的 lodash,因此 将通用 lodash requires
转换为特定需要的功能 ,以避免代码重复。这可能会为你节省相当多的 JavaScript 负载。
Shubham Kanodia 撰写了一份 详细的关于智能打包的低维护指南 :如何在生产环境中实现仅仅将遗留代码推送到老版本浏览器上,里面还有一些你可以直接拿来用的代码片段。
Jake Archibald 发布了一篇详细的文章,其中包含了 需要牢记的 ES 模块相关内容 ,例如:内联脚本会被推迟,直到正在阻塞的外部脚本和内联脚本得到执行。(预览大图)
30. 是否使用了 JavaScript 差异化服务?
我们希望通过网络发送必要的 JavaScript,但这意味着需要更加集中精力并且细粒度地关注这些静态资源的传送。前一阵子 Philip Walton 介绍了差异化服务的想法。该想法是编译和提供两个独立的 JavaScript 包:“常规”构建,带有 Babel-transforms 和 polyfill 的构建,只提供给实际需要它们的旧浏览器,以及另一个没有转换和 polyfill 的包(具有相同功能)。
结果,通过减少浏览器需要处理的脚本数量来帮助减少主线程的阻塞。Jeremy Wagner 在 2019 年发布了一篇 关于差异服务以及如何在你的构建管道中进行设置的综合文章 ,从设置 babel 到你需要在 webpack 中进行哪些调整,以及完成所有这些工作的好处。
31. 通过增量解耦识别和重写遗留代码
老项目充斥着陈旧和过时的代码。重新查看你的依赖项,评估重构或重写最近导致问题的遗留代码所需的时间。当然,它始终是一项重大任务,但是一旦你了解了遗留代码的影响,就可以从增量解耦开始。
首先,设置指标,跟踪遗留代码调用的比率是保持不变或是下降,而不是上升。公开阻止团队使用该库,并确保你的 CI 能够 警告 开发人员,如果它在拉取请求(pull request)中使用。Polyfill 可以帮助将遗留代码转换为使用标准浏览器功能的重写代码库。
32. 识别并删除未使用的 CSS/JS
Chrome 中的 CSS 和 JavaScript 代码覆盖率 可以让你了解哪些代码已执行/已应用,哪些代码尚未执行。你可以开始记录覆盖范围,在页面上执行操作,然后浏览代码覆盖率结果。一旦你检测到未使用的代码, 找到那些模块并使用 import()
延迟加载 (参见整个线程)。然后重复覆盖配置文件并验证它现在在初始加载时发送的代码是否变少了。
你可以使用 Puppeteer 以 编程方式收集代码覆盖率 ,Canary 也能够让你导出代码覆盖率结果。正如 Andy Davies 提到的那样,你可能希望 同时收集现代和旧版浏览器 的代码覆盖率。 Puppeteer 还有许多其他用例 ,例如,自动视差或 监视每个构建的未使用的 CSS 。
此外, purgecss 、 UnCSS 和 Helium 可以帮助你从 CSS 中删除未使用的样式。如果你不确定是否在某处使用了可疑的代码,可以遵循Harry Roberts 的建议:为该 class 创建 1×1px 透明 GIF 并将其放入 dead/
目录,例如: /assets/img/dead/comments.gif
。然后,将该特定图像设置为 CSS 中相应选择器的背景,然后静候几个月,查看该文件能否出现在你的日志中。如果日志里没出现该条目,则没有人使用该遗留组件:你可以继续将其全部删除。
对于爱冒险的人,你甚至可以通过使用 DevTools 监控 DevTools ,通过一组页面自动收集未使用的 CSS。
33. 减小 JavaScript 包的大小
正如 Addy Osmani指出的那样,当你只需要一小部分时,你很可能会发送完整的 JavaScript 库,以及提供给不需要它们的浏览器的过时 polyfill,或者只是重复代码。为避免额外开销,请考虑使用 webpack-libs-optimization ,在构建过程中删除未使用的方法和 polyfill。
将打包审计添加到常规工作流程中。有一些你在几年前添加的重型库的轻量级替代品,例如:Moment.js 可以用 date-fns 或Luxon 代替。Benedikt Rötsch 的研究表明,从 Moment.js 到 date-fns 的转换可能会使 3G 和低端手机上的首次绘制时间减少大约 300ms。
这就是Bundlephobia 这样的工具可以帮助你找到在程序包中添加 npm 包的成本。你甚至可以 将这些成本与 Lighthouse Custom Audit 相结合 。这也适用于框架。通过删除或减小Vue MDC 适配器(Vue 的 Material 组件),样式可以从 194KB 降至 10KB。
喜欢冒险吗?你可以看看 Prepack 。它将 JavaScript 编译为等效的 JavaScript 代码,但与 Babel 或 Uglify 不同,它允许你编写正常的 JavaScript 代码,并输出运行速度更快的等效 JavaScript 代码。
除了传送整个框架包之外,你甚至可以修剪框架并将其编译为不需要额外代码的原始 JavaScript 包。Svelte 做到了, Rawact Babel 插件 也是如此,它在构建时将 React.js 组件转换为原生 DOM 操作。 为什么?好吧,正如维护者解释的那样:“React-dom 包含可以渲染的每个可能组件/ HTMLElement 的代码,包括用于增量渲染、调度、事件处理等的代码。但是有些应用程序不需要所有这些功能(在初始页面加载时)。对于此类应用程序,使用原生 DOM 操作构建交互式用户界面可能是有意义的。”
在 Benedikt Rötsch 的文章中 ,他表示,从 Moment.js 到 date-fns 的转换会使 3G 和低端手机上的首次绘制时间减少大约 300ms。(预览大图)
34. 是否使用了 JavaScript 代码块的预测预获取?
我们可以使用启发式方法来决定何时预加载 JavaScript 代码块。 Guess.js 是一组工具和库,它使用 Google Analytics 的数据来确定用户最有可能从给定页面访问哪个页面。根据从 Google Analytics 或其他来源收集的用户导航模式,Guess.js 构建了一个机器学习模型,用于预测和预获取每个后续页面中所需的 JavaScript。
因此,每个交互元素都接收参与的概率评分,并且基于该评分,客户端脚本决定提前预获取资源。你可以将该技术集成到 Next.js 应用程序、Angular 和 React 中,还有一个 webpack 插件 能够自动完成设置过程。
显然,你可能会让浏览器预测到使用不需要的数据从而预获取到不需要的页面,因此最好在预获取请求的数量上保持绝对保守。一个好的用例是预获取结账中所需的验证脚本,或者当一个关键的 CTA(call-to-action)进入视口时的推测性预获取。
需要不太复杂的东西? Quicklink 是一个小型库,可在空闲时自动预获取视口中的链接,以便加快下一页导航的加载速度。但是,它也考虑了数据流量,因此它不会在 2G 网络或者 Data-Saver
打开时预获取数据。
35. 从针对你的目标 JavaScript 引擎进行优化中获得好处
研究哪些 JavaScript 引擎在你的用户群中占主导地位,然后探索针对这些引擎的优化方法。例如,在为 Blink 内核浏览器、Node.js 运行时和 Electron 中使用的 V8 进行优化时,使用脚本流来处理庞大的脚本。它允许在下载开始时在单独的后台线程上解析 async
或 defer scripts
,因此在某些情况下可以将页面加载时间减少多达 10%。实际上,在 <head>
里 使用 <script defer>
,以便浏览器可以提前发现资源,然后在后台线程上解析它。
警告: Opera Mini不支持脚本延迟,所以如果你正在为印度或非洲开发 , defer
将被忽略,这会导致阻止渲染,直到脚本执行完为止(感谢 Jeremy!) 。
渐进式启动意味着使用服务器端渲染来获得快速的首次有效绘制,但也包括一些最小的 JavaScript,以保持首次交互时间接近首次有效绘制时间。
36. 使用客户端渲染还是服务器端渲染?
在这两种情况下,我们的目标应该是设置渐进式启动:使用服务器端渲染来获得快速的首次有效绘制,但也包括一些最小的必要 JavaScript,以保持首次交互时间接近首次有效绘制时间。如果 JavaScript 在首次有效绘制之后来得太晚,浏览器可能会在解析、编译和执行后期发现的 JavaScript 时锁定主线程,从而给站点或应用程序的交互带来枷锁。
为避免这种情况,请始终将函数执行分解为独立的异步任务,并尽可能使用 requestIdleCallback
。考虑使用 webpack 的 动态 import()
支持 ,延迟加载 UI 的部分,降低加载、解析和编译成本,直到用户真正需要它们( 感谢 Addy! )。
从本质上讲,首次可交互时间(TTI)告诉我们导航和交互之间的时间。通过查看初始内容渲染后的前五秒窗口来定义度量标准,其中任何 JavaScript 任务都不会超过 50 毫秒。如果发生超过 50 毫秒的任务,则重新开始搜索五秒钟窗口。因此,浏览器将首先假设它已到达交互状态,然后切换到冻结状态,最终切换回交互状态。
一旦我们到达交互状态,在按需或在时间允许的情况下,就可以启动应用程序的非必要部分。不幸的是,正如 Paul Lewis 所注意到的那样 ,框架通常没有提供给开发者优先级的概念,因此大多数库和框架都难以实现渐进式启动。如果你有时间和资源,请使用此策略最终提升性能。
那么,客户端还是服务器端?如果用户没有明显的好处, 客户端渲染可能不是真正必要的 —— 实际上,服务器端渲染的 HTML 可能更快。也许你甚至可以 使用静态站点生成器预渲染一些内容 ,并将它们直接推送到 CDN,并在顶部添加一些 JavaScript。
将客户端框架的使用限制为绝对需要它们的页面。如果做得不好,服务器渲染和客户端渲染是一场灾难。考虑在 构建时预渲染 和 动态 CSS 内联 ,以生成生产就绪的静态文件。Addy Osmani 就可能值得关注的 JavaScript 成本发表了精彩的演讲 。
37. 约束第三方脚本的影响
通过所有性能优化,我们通常无法控制来自业务需求的第三方脚本。第三方脚本指标不受最终用户体验的影响,因此通常一个脚本最终会调用令人讨厌的冗长的第三方脚本,从而破坏了专门的性能工作。为了控制和减轻这些脚本带来的性能损失,仅仅异步加载它们(可能是通过延迟)并通过资源提示(如 dns-prefetch
或 preconnect
)加速它们是不够的。
正如 Yoav Weiss 在他 关于第三方脚本的必读观点 中所解释的那样,在许多情况下,这些脚本会下载动态的资源。资源在页面加载之间发生变化,因此我们没有必要知道从哪些主机下载资源以及这些资源是什么。
你有哪些选择方案?考虑 使用 service worker,通过超时竞争资源下载 ,如果资源在特定超时内没有响应,则返回空响应以告知浏览器继续解析页面。你还可以记录或阻止未成功或不符合特定条件的第三方请求。如果可以,请从你自己的服务器而不是从供应商的服务器加载第三方脚本。
Casper.com 发布了一个详细的案例研究,说明他们如何通过自托管的 Optimizely 网站响应时间减少了 1.7 秒。这可能是值得的。(图片来源)( 预览大图 )
另一种选择是建立 内容安全策略 (CSP)以限制第三方脚本的影响,例如:不允许下载音频或视频。最好的选择是通过 <iframe>
嵌入脚本,以便脚本在 iframe 的上下文中运行,因此第三方脚本无法访问页面的 DOM,也无法在你的域上运行任意代码。使用 sandbox
属性可以进一步约束 iframe,那样你就可以禁用一切 iframe 可能执行的任何功能,例如:防止脚本运行、阻止警报、表单提交、插件、访问顶部导航等。
比如,可能必须使用 <iframe sandbox="allow-scripts">
来运行脚本。每个限制都可以通过 sandbox
属性上的各种 allow
值来解除( 几乎所有的浏览器都受支持 ),因此将它们限制在应该允许的最低限度。
考虑使用 Intersection Observer;这将使广告仍然在 iframe 中,但是可以调度事件或从 DOM 获取所需信息(例如,广告可见性)。可以关注一些新的策略,例如功能策略,资源大小限制和 CPU/带宽优先级,以限制可能会降低浏览器速度的有害 Web 功能和脚本,例如:同步脚本、同步 XHR 请求、 document.write
和过时的实现。
要对第三方进行压力测试,请检查 DevTools 中性能配置文件页面中的自下而上的摘要,测试如果请求被阻止或超时的情况会发生什么 —— 对于后者,你可以使用 WebPageTest 的 Blackhole 服务器 blackhole.webpagetest.org
,它可以将特定域指向你的 hosts
文件。最好是 自托管并使用单一主机名 ,但也可以生成一个请求映射,该映射公开第四方调用并检测脚本何时更改。你可以使用 Harry Roberts 的方法审核第三方,并生成类似这样的电子表格。Harry 还在他 关于第三方性能和审计的讨论中 解释了审计工作流程。
图片来源:Harry Roberts
38. 设置 HTTP 缓存标头
仔细检查是否已正确设置 expires
、 max-age
、 cache-control
和其他 HTTP 缓存头。通常,资源无论在 短时间内(如果它们可能会更改)还是无限期(如果它们是静态的) 情况下都是可缓存的 —— 你只需在需要时在 URL 中更改它们的版本。禁用 Last-Modified
标头,因为任何带有它的静态资源都将导致带有 If-Modified-Since
标头的条件请求,即使资源位于缓存中也是如此。 Etag
也是如此。
使用使用专为指纹静态资源设计的 Cache-control:immutable
,以避免重新验证(截至 2018 年 12 月, Firefox、Edge 和 Safari 都已经支持该功能 ; Firefox 仅支持 https://
事务)。事实上,“在 HTTP 存档中的所有页面中,2% 的请求和 30% 的网站似乎 包含至少 1 个不可变响应 。此外,大多数使用它的网站都设置了具有较长新鲜生命周期的静态资源。”
还记得 stale-while-revalidate 吗?你可能知道,我们使用 Cache-Control
响应头指定缓存时间,例如: Cache-Control: max-age=604800
。经过 604800 秒后,缓存将重新获取所请求的内容,从而导致页面加载速度变慢。通过使用 stale-while-revalidate
可以避免这种速度变慢的问题。它本质上定义了一个额外的时间窗口,在此期间缓存可以使用旧的静态资源,只要它在异步地在后台重新验证自己。因此,它“隐藏了”来自客户端的延迟(在网络和服务器上)。
在 2018 年 10 月,Chrome 发布了一个意图 在 HTTP Cache-Control 标头中对 stale-while-revalidate
的处理,因此,它应该会改善后续页面加载延迟,因为旧的静态文件不再位于关键路径中。结果: 重复访问页面的 RTT 为零 。
你可以使用 Heroku 的 HTTP 缓存标头入门 ,Jake Archibald 的“缓存最佳实践”和Ilya Grigorik 的HTTP 缓存入门作为指南。另外,要注意标头的变化,特别是与 CDN 相关的标头,并注意Key 标头,这有助于避免当新请求与先前请求略有差异(但不显着)时,需要进行额外的往返验证( 感谢 Guy! )。
另外,请仔细检查你是否发送了不必要的标头(例如 x-powered-by
、 pragma
、 x-ua-compatible
、 expires
等),并且包含有用的安全性和性能标头(例如 Content-Security-Policy
, X-XSS-Protection
, X-Content-Type-Options
等)。最后,请记住单页应用程序中CORS 请求的性能成本。
- [译] 2019 前端性能优化年度总结 — 第一部分
- [译] 2019 前端性能优化年度总结 — 第二部分
- [译] 2019 前端性能优化年度总结 — 第三部分
- [译] 2019 前端性能优化年度总结 — 第四部分
- [译] 2019 前端性能优化年度总结 — 第五部分
- [译] 2019 前端性能优化年度总结 — 第六部分
如果发现译文存在错误或其他需要改进的地方,欢迎到 掘金翻译计划 对译文进行修改并 PR,也可获得相应奖励积分。文章开头的 本文永久链接 即为本文在 GitHub 上的 MarkDown 链接。
掘金翻译计划 是一个翻译优质互联网技术文章的社区,文章来源为掘金 上的英文分享文章。内容覆盖 Android 、 iOS 、 前端 、 后端 、 区块链 、 产品 、 设计 、 人工智能 等领域,想要查看更多优质译文请持续关注 掘金翻译计划 、官方微博、 知乎专栏 。
以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 码农网
猜你喜欢:本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
精通CSS(第2版)
[英] Andy Budd、[英] Simon Collison、[英] Cameron Moll / 陈剑瓯 / 人民邮电出版社 / 2010-5 / 49.00元
本书汇集了最有用的CSS技术,介绍了CSS的基本概念和最佳实践,结合实例探讨了图像、链接和列表的操纵,还有表单设计、数据表格设计、纯CSS布局等核心CSS技术。此外,书中着眼于创建跨浏览器的技术,讨论了bug及其捕捉和修复技术,还将所有技术组合成两个精彩的实例,讲述这些技术的工作原理和实际用法。 本书适合具有HTML和CSS基础知识的读者阅读。一起来看看 《精通CSS(第2版)》 这本书的介绍吧!