大家好,我是ConardLi。


(资料图)

说到HTTP​的103状态码,你可能很早就听说过了,但是你不一定真的理解了它。

这很正常,这个状态码早在2017年就被提出来了,但是支持它的服务器和浏览器真的很少。

直到前几天,Chrome​宣布在Chrome 103​版本对HTTP 103状态码提供了支持,不得不说老外还挺皮啊...

今天我们就来看一下,HTTP 103状态码究竟有什么用途。

资源加载的性能问题

随着时间的推移,网站变得越来越复杂。一些大型网站的服务器可能需要执行很多重要的工作(例如,访问数据库或访问源服务器的CDN​)来为请求的页面生成HTML。

但是,这种服务器的思考时间​会在浏览器开始渲染页面之前带来额外的延迟。因为浏览器需要先把HTML​页面加载回来,才能知道下一步去加载哪些JavaScript、CSS或字体文件等。中间这段时间实际上就浪费掉了,对用户访问我们的页面来讲,这段等待时间就是白屏或是不可用的状态。

我们来看看抖音Web​站的资源加载:浏览器先要等待前面两个HTML​的大约800 ms​的时间才能去加载后面的JS 、CSS等资源文件。

有没有办法在等待HTML响应的同时就去提前把重要的静态资源文件也加载回来呢?

HTTP 103 状态码

HTTP 103​状态码 (Early Hints​) 是一个信息性HTTP​状态代码,可以用于在最终响应之前发送一个初步的HTTP响应。

利用HTTP 103​状态码,就可以让服务器在服务器处理主资源的同时向浏览器发送一些关键子资源(JavaScript、CSS或字体文件)或页面可能使用的其他来源的提示。

浏览器可以使用这些提示来预热连接,并在等待主资源响应的同时请求子资源。换句话说,Early Hints​可以通过提前做一些工作来帮助浏览器利用这种服务器思考时间,从而提升页面的渲染性能。

在某些情况下,这可以帮助LCP​(最大内容绘制)至少提升几百毫秒。例如在Shopify​和Cloudflare​所观察到的来看,LCP大概提升了 1 秒。

启用 Early Hints 前后对比

什么样的网站适合 Early Hints

在开始使用之前,可能要先思考下,什么样的网站比较适合这个优化。

如果你的网站的主页面响应非常快,可能没什么必要。比如对于大部分SPA(单页应用),可能用处不是那么大。

在SPA​中,大部分的逻辑都在客户端,HTML​很小,下发HTML​的服务器也基本就是没有什么逻辑的静态服务器。大部分情况下只会包括一个Root​节点,以及一些资源的Link​,大部分逻辑和加载时间其实都在打包后的JavaScript​中。这种情况我们只需要使用常规的rel=preload、rel=preconnect等手段就可以了。

但是在SSR​项目中,加载HTML​往往需要在服务端花费更多的时间,因为服务端可能和数据库交互以及将数据拼接成HTML​元素。相比之下,加载其他的脚本和样式资源可能花费的时间要更短一点,这种站点启用Early Hints是比较合适的。

启用 Early Hints

在Chrome 103​版本,对HTTP 103​状态码 (Early Hints) 提供了支持。

启用Early Hints​的第一步就是要确认我们站点的主页面​,也就是用户通常在访问我们的网站时开始的页面。如果我们有很多来自其他网站的用户,主页面可能就是主页或热门的产品列表页面。

Early Hints的用途会随着用户在我们的站内导航的次数而降低,因为浏览器可能已经在前几次导航中把所有需要的子资源请求回来了,给用户良好的第一印象是最重要的!

确认了站点的主页面​,下一步就是确定哪些来源或子资源将是最佳预连接或预加载的候选者。通常情况家,我们要找的就是对关键用户指标(LCP​或FP​)贡献最大的源和子资源。具体一点,就是找到阻塞渲染的子资源,例如同步JavaScript、样式表,甚至网络字体等。

然后就是尽量避免选择已经过时或者不再被主页面使用的资源。

比如一些经常更新或者带有hash​的资源(conardli.top/static/home.aaaa1.js),尽量不要选择,这可能会导致页面需要加载的资源和实际预加载的资源不匹配。

比如我们举个例子:

首先我们去服务器请求主页面:

GET /home.htmlHost: conardli.topUser-Agent: [....] Chrome/103.0.0.0 [...]

服务器预测站点将需要home.aaaa1.js​,并建议通过Early Hints预加载它:

103 Early HintsLink: ; rel=preload; as=script[...]

随后,客户端马上向服务端请求了home.aaaa1.js​。然而,这时JS​资源更新了,主页面已经需要另外一个版本的JS了。

200 OK[...] Conardli"s Blog

所以,我们最好选择一些比较稳定的资源进行预加载,我们可以对 JS 进行拆包,将不经常发生变化的稳定部分和经常发生更新的动态脚本部分拆成多个资源。我们只对稳定部分实施预加载,在浏览器获取到主页面后再去加载动态部分。

code秘密花园

最后,在服务器端,查找已知支持Early Hints​的浏览器发送的主页面请求,并响应103 Early Hints​。在103​响应中,会包括相关的预连接和预加载提示。主页面准备好后,再按照正常的响应进行响应。为了向后兼容,最好在最终响应中包含LINK HTTP标头,甚至也可以增加在生成主页面时需要的一些明显的关键资源。

Early Hints响应:

GET /main.htmlHost: conardli.topUser-Agent: [....] Chrome/103.0.0.0 [...]103 Early HintsLink: ; rel=preconnectLink: ; rel=preload; as=styleLink: ; rel=preload; as=script

成功响应:

200 OKContent-Length: 7531Content-Type: text/html; charset=UTF-8Content-encoding: brLink: ; rel=preconnectLink: ; rel=preload; as=styleLink: ; rel=preload; as=scriptLink: ; rel=preload; as=style code秘密花园 <script src="/common.js"></script> 和 HTTP2/Push 有什么关系?

看到这里你可能发现了,这玩意和HTTP2​的服务器推送 (Server Push) 很像啊。

Server Push​即在浏览响应HTML文件的时候,服务器会同时将所需的资源文件主动推送给浏览器。

浏览器在收到推送的资源之后会缓存到本地。等解析HTML发现需要加载对应资源的时候会直接从本地读取,不必再等待网络传输了。

虽然这听起来很神奇,但这个方案有非常大的缺陷:Server Push​很难避免推送浏览器已经拥有的子资源,其实很多资源在浏览器第一次请求到就已经缓存下来了。这种 “过度推动” 会导致网络带宽的使用效率降低,从而显着阻碍性能优势。总体而言,Chrome​数据显示HTTP2/Push实际上对整个网络的性能产生了负面影响。

所以,Chrome​宣布移除了对HTTP/2 Server Push特性的支持:

相比之下,Early Hints​在实践中表现更好,因为它结合了发送初步响应的能力和提示,浏览器实际上只负责获取它实际需要的资源。虽然Early Hints​还没有涵盖HTTP/2 Server Push​理论上可以解决的所有用例,但是它解决了网络带宽浪费的问题,可以说是HTTP/2 Server Push的升级版。

支持情况

浏览器支持情况:

服务器支持情况:

Node.js​:通过Fastify插件提供支持;Apache​:通过mod_http2支持;H2O:支持;Nginx:实验模块。

标签: HTTP HTTP