做者簡介 felix 螞蟻金服·數據體驗技術團隊javascript
本文主要介紹preload的使用,以及與prefetch的區別。而後會聊聊瀏覽器的加載優先級。css
preload 提供了一種聲明式的命令,讓瀏覽器提早加載指定資源(加載後並不執行),在須要執行的時候再執行。提供的好處主要是html
<!-- 使用 link 標籤靜態標記須要預加載的資源 -->
<link rel="preload" href="/path/to/style.css" as="style">
<!-- 或使用腳本動態建立一個 link 標籤後插入到 head 頭部 -->
<script> const link = document.createElement('link'); link.rel = 'preload'; link.as = 'style'; link.href = '/path/to/style.css'; document.head.appendChild(link); </script>
複製代碼
Link: <https://example.com/other/styles.css>; rel=preload; as=style
複製代碼
如咱們經常使用到的 antd 會依賴一個 CDN 上的 font.js 字體文件,咱們能夠設置爲提早加載,以及有一些模塊雖然是按需異步加載,但在某些場景下知道其一定會加載的,則能夠設置 preload 進行預加載,如:java
<link rel="preload" as="font" href="https://at.alicdn.com/t/font_zck90zmlh7hf47vi.woff">
<link rel="preload" as="script" href="https://a.xxx.com/xxx/PcCommon.js">
<link rel="preload" as="script" href="https://a.xxx.com/xxx/TabsPc.js">
複製代碼
目前咱們支持的瀏覽器主要爲高版本 Chrome,因此可放心使用 preload 技術。 其餘環境在 caniuse.com 上查到的支持狀況以下: git
在不支持 preload 的瀏覽器環境中,會忽略對應的 link 標籤,而若須要作特徵檢測的話,則:const isPreloadSupported = () => {
const link = document.createElement('link');
const relList = link.relList;
if (!relList || !relList.supports) {
return false;
}
return relList.supports('preload');
};
複製代碼
preload 是確認會加載指定資源,如在咱們的場景中,x-report.js 初始化後必定會加載 PcCommon.js 和 TabsPc.js, 則能夠預先 preload 這些資源;github
prefetch 是預測會加載指定資源,如在咱們的場景中,咱們在頁面加載後會初始化首屏組件,當用戶滾動頁面時,會拉取第二屏的組件,若能預測用戶行爲,則能夠 prefetch 下一屏的組件。web
使用 preload 前,在遇到資源依賴時進行加載: chrome
使用 preload 後,無論資源是否使用都將提早加載: 能夠看到,preload 的資源加載順序將被提早:使用 preload 後,Chrome 會有一個警告: 跨域
如上文所言,若不肯定資源是一定會加載的,則不要錯誤使用 preload,以避免本末倒置,給頁面帶來更沉重的負擔。瀏覽器
固然,能夠在 PC 中使用 preload 來刷新資源的緩存,但在移動端則須要特別慎重,由於可能會浪費用戶的帶寬。
preload 和 prefetch 混用的話,並不會複用資源,而是會重複加載。
<link rel="preload" href="https://at.alicdn.com/t/font_zck90zmlh7hf47vi.woff" as="font">
<link rel="prefetch" href="https://at.alicdn.com/t/font_zck90zmlh7hf47vi.woff" as="font">
複製代碼
使用 preload 和 prefetch 的邏輯可能不是寫到一塊兒,但一旦發生對用一資源 preload 或 prefetch 的話,會帶來雙倍的網絡請求,這點經過 Chrome 控制檯的網絡面板就能甄別:
若 css 中有應用於已渲染到 DOM 樹的元素的選擇器,且設置了 @font-face 規則時,會觸發字體文件的加載。 而字體文件加載中時,DOM 中的這些元素,是處於不可見的狀態。對已知必加載的 font 文件進行預加載,除了有性能提高外,更有體驗優化的效果。
在咱們的場景中,已知 antd.css 會依賴 font 文件,因此咱們能夠對這個字體文件進行 preload:
<link rel="preload" as="font" href="https://at.alicdn.com/t/font_zck90zmlh7hf47vi.woff">
複製代碼
然而我發現這個文件加載了兩次:
緣由是對跨域的文件進行 preload 的時候,咱們必須加上 crossorigin 屬性:
<link rel="preload" as="font" crossorigin href="https://at.alicdn.com/t/font_zck90zmlh7hf47vi.woff">
複製代碼
再看一下網絡請求,就變成一條了。
W3 規範是這麼解釋的:
Preload links for CORS enabled resources, such as fonts or images with a crossorigin attribute, must also include a crossorigin attribute, in order for the resource to be properly used.
那爲什麼會有兩條請求,且優先級不一致,又沒有命中緩存呢?這就得引出下一個話題來解釋了。
咱們先來看一張圖:
這張表詳見:Chrome Resource Priorities and Scheduling
這張圖表示的是,在 Chrome 46 之後的版本中,不一樣的資源在瀏覽器渲染的不一樣階段進行加載的優先級。 在這裏,咱們只須要關注 DevTools Priority 體現的優先級,一共分紅五個級別:
第二條 Highest 也就是樣式引入的請求:
能夠看到,在 preload 的請求中,缺乏了一個 origin 的請求頭字段,表示這個請求是匿名的請求。 讓這兩個請求能共用緩存的話,目前的解法是給 preload 加上 crossorigin 屬性,這樣請求頭會帶上 origin, 且與樣式引入的請求同源,從而作到命中緩存:
<link rel="preload" as="font" crossorigin href="https://at.alicdn.com/t/font_zck90zmlh7hf47vi.woff">
複製代碼
這麼請求就只剩一個:
在網絡瀑布流圖中,也顯示成功預加載且後續命中緩存再也不二次加載:preload 是個好東西,能告訴瀏覽器提早加載當前頁面必須的資源,將加載與解析執行分離開,作得好能夠對首次渲染帶來不小的提高,但要避免濫用,區分其與 prefetch 的關係,且須要知道 preload 不一樣資源時的網絡優先級差別。
preload 加載頁面必需的資源如 CDN 上的字體文件,與 prefetch 預測加載下一屏數據,興許是個不錯的組合。
參考資料:
團隊提早給你們拜個早年,感謝你們對於專欄的支持~對團隊感興趣的同窗能夠關注專欄或者發送簡歷至'tao.qit####alibaba-inc.com'.replace('####', '@'),歡迎有志之士加入~