聊一聊 webpack 中的 preloading 和 Prefetching

来源:https://www.cnblogs.com/zhangguicheng/archive/2019/12/21/12077364.html
-Advertisement-
Play Games

聊一聊 webpack 中的 preloading 和 Prefetching 提到 Preloading 和 Prefetching 就不得不先說一下代碼分割,通過下麵的例子我們來說明為什麼需要代碼分割? 在首次訪問時, index.js 文件的大小為 2 MB,需要載入的大小是 2 MB 業務代 ...


聊一聊 webpack 中的 preloading 和 Prefetching

提到 Preloading 和 Prefetching 就不得不先說一下代碼分割,通過下麵的例子我們來說明為什麼需要代碼分割?

// index.js
import _ from 'lodash'; // 假設大小為 1 MB

業務代碼 // 假設大小為 1 MB
  • 在首次訪問時, index.js 文件的大小為 2 MB,需要載入的大小是 2 MB
  • 業務代碼改變用戶再次訪問時,index.js 的大小為 2 MB,需要載入的大小還是 2 MB

現在進行代碼分割:

// src/index.js 

業務代碼 // 假設大小為 1 MB

// src/lodash.js
import _ from 'lodash';
window._ = _; // 以後在其它文件中使用 _ 就可以使用 lodash 庫了。
  • 首次訪問時,index.js 1 MB,lodash.js 1 MB , 需要載入的大小是 2 MB,而且此時可以進行並行載入,速度一般會比上面的快。

  • 業務代碼改變用戶再次訪問時,index.js 1 MB,由於 lodash.js 文件並沒有發生變化,所以無需再次載入,因為瀏覽器的緩存中有,所以此次只需載入 1 MB。

從上面的例子可以看出,代碼分割提高了性能,但是第一次訪問的時間並沒有減少多少,webpack 想讓第一次訪問的時候也得到很大的優化。

我們先從 webpack 中的 SplitChunkPlugin 的預設配置中找到答案,

optimazition: {
  splitChunks: {
    chunks: 'async', // 非同步代碼才會進行代碼分割
    ...
  }
}

我們可以看到,chunks 的配置是 async ,只有當非同步時才會進行代碼分割。

webpack 為什麼要這樣預設設置呢?

還是從下麵的例子來說明:

創建一個div元素,併在頁面上顯示出來。

// index.js
document.addEventListener('click', () => {
  const div = document.createElement('div');
  div.innerHTML = 'hello webpack';
  document.body.appendChild(div);
});

思考上面的代碼寫的有問題嗎?還有優化的空間嗎?

現在我們將上面的代碼打包在瀏覽器中運行,在瀏覽器中 按 Ctrl + Shift + P ,然後在彈出的對話框中輸入 coverage ,點擊回車,然後再點擊下麵的小黑原點,小黑圓點變成紅圓點之後,刷新頁面,會出現下圖所示的頁面:

coverage example

從紅色的方框中可以看出當前載入的文件中在當前頁面中的利用率為 74.6%

仔細分析一下上面的代碼,在回調函數中的下麵三行代碼只有在點擊頁面之後才會有用,因此載入頁面時沒必要載入它們。

const div = document.createElement('div');
div.innerHTML = 'hello webpack';
document.body.appendChild(div);

現在我們換一種寫法,將它們非同步載入進來,現在新建一個 click 文件:

// click.js
function handleClick() {
  const div = document.createElement('div');
  div.innerHTML = 'hello webpack';
  document.body.appendChild(div);
}

export default handleClick;

然後改寫 index.js 文件:

document.addEventListener('click', () => {
  import('./click.js').then(({default: func}) => {
    func();
  })
});

這時候將非同步代碼寫在一個單獨的文件中,只有當點擊頁面時才會去載入 click.js 這個文件。

現在再看此時的代碼利用率為 75% 有了一點提升,設想如果非同步載入在的代碼比較大的話,提升的會比較多。

改變代碼後coverage

現在我們就看出來 webopack 為什麼要使用 chunks: 'async' 這樣的預設配置了。

webpack 優化的側重點是代碼的使用率而不是緩存,只是使用緩存的方式來優化意義是不大的,通過非同步的方式提高代碼的利用率才能比較大程度地提高網站的性能。

這也是為什麼老提倡寫非同步代碼的原因。

現在又有一個問題,只有當用戶點擊頁面時才會載入 click.js這個文件,那麼如果這個文件很大,那載入的時間也會很長呀,用戶體驗也不高呀。

那這個問題應該如何解決呢?

有些小伙伴可能會想,能不能在載入完頁面網路空閑的時候先把這些文件載入進來呀,真聰明,這就是接下來要講的 Preloading 和 Prefetching。

  • Prefetching

    使用方法也比較簡單,就是在要非同步載入的文件前面加上 /* webpackPrefetch: true */ 這個 magic comment 即可。

    document.addEventListener('click', () => {
      import(/* webpackPrefetch: true */ './click.js').then(({default: func}) => {
        func();
      })
    });

    Prefetching

    上圖中的 0.js 是 click.js 打包之後,可以看出在頁面載入完之後的空閑時間還沒有點擊頁面時已經載入了 0.js ,當點擊頁面時,0.js 直接從緩存中讀取,因此耗時非常短。

  • Preloading 和 Prefetching 有什麼區別?

    兩者的最大區別在於,Prefetching 是在核心代碼載入完成之後帶寬空閑的時候再去載入,而 Preloading 是和核心代碼文件一起去載入的。

因此,使用 Prefetching 的方式去載入非同步文件更合適一些,不過要註意瀏覽器的相容性問題。

完, 如有不恰當之處,歡迎指正哦。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 案例:開機動畫 由上下兩部分組成,先下麵的高變為0 ,再最大的div寬為0,形成一個縮小到沒有的動畫效果 點擊的X是在背景圖上的,在上面設置了一個空的span用於註冊點擊事件 <!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8"> ...
  • 案例:手風琴 封裝好的動畫函數在common.js裡面 //function getStyle(element, attr) {...} //function animate(element, json, fn) {...} 手風琴設置的是背景圖嗎,backgroudImage="url(image ...
  • 圖文結合深入理解 JS 中的 this 值 在 中最常見的莫過於函數了,在函數(方法)中 的出現頻率特別高,那麼 到底是什麼呢,今天就和大家一起學習總結一下 中的 。 1. 初探this 在 中是一個關鍵字,不是變數也不是屬性名, 中不允許給this賦值。 它是函數運行時,在函數體內部自動生成的一個 ...
  • 聊一聊 JS 輸出為 [object object] 是怎麼回事? 今天在學習ES6中的 數據類型時,在寫demo時控制台輸出為 ,當時有點疑惑,查閱了相關資料後搞清楚了其中的原因。 在解釋之前,由於有些小伙伴可能還沒有接觸過ES6,所以先說一下上面用到的ES6的一些特性: const: 聲明一個常 ...
  • 接續上篇ES6+轉ES5,本篇將使用webpack和babel將多個不同目錄下指定的多個ES6+語法的js文件編譯為ES5,並將編譯後的文件配置註入對應的html文件。 所需環境node、npm、設置淘寶鏡像請參考上篇進行安裝,地址:https://www.cnblogs.com/puyongson ...
  • 1.關於浮動 浮動的元素會脫離標準文檔流(float),從而不占據空間,實現了一行排列多個元素的效果 ,但是又導致上級元素height消失,處理這種情況的方法就是有兩種: 1.第一種在css里寫個偽類,哪些地方需要清除浮動,直接調用類名。 .clearfix::after{ display: blo ...
  • 基於jquery的提示框JavaScript 插件,類Bootstrap [TOC] 源碼 github地址: "https://github.com/Ethan Xie/message" 實例 通過此此插件可以為提示欄添加自動/點擊消失的功能 用法 需引入message.css與message.j ...
  • 第一步: 在 VSCode 中,安裝用於同步配置的插件 settings sync 第二步:將 VSCode 配置上傳到 GitHub 完成這一步需要 GitHub token 和 GitHub gist 進入GitHub 設置界面 & 創建 GitHub token 在這裡找到之前你上傳 VSCo ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...