Webpack學(xué)習(xí)之動態(tài)import原理及源碼分析
前言
在開始之前,先給我的mini-react打個廣告。對react源碼感興趣的朋友,走過路過的朋友點個star
在平時的開發(fā)中,我們經(jīng)常使用 import()實現(xiàn)代碼分割和懶加載。在低版本的瀏覽器中并不支持動態(tài) import(),那 webpack 是如何實現(xiàn) import() polyfill 的?
原理分析
我們先來看看下面的 demo
function component() { const btn = document.createElement("button"); btn.onclick = () => { import("./a.js").then((res) => { console.log("動態(tài)加載a.js..", res); }); }; btn.innerHTML = "Button"; return btn; } document.body.appendChild(component());
點擊按鈕,動態(tài)加載 a.js
腳本,查看瀏覽器網(wǎng)絡(luò)請求可以發(fā)現(xiàn),a.js
請求返回的內(nèi)容如下:
簡單看,實際上返回的就是下面這個東西:
(self["webpackChunkwebpack_demo"] = self["webpackChunkwebpack_demo"] || []).push([ ["src_a_js"], { "./src/a.js": () => {}, }, ]);
從上面可以看出 3 點信息:
- 1.webpackChunkwebpack_demo 是掛到全局 window 對象上的屬性
- 2.webpackChunkwebpack_demo 是個數(shù)組
- 3.webpackChunkwebpack_demo 有個 push 方法,用于添加動態(tài)的模塊。當(dāng)
a.js
腳本請求成功后,這個方法會自動執(zhí)行。
再來看看 main.js 返回的內(nèi)容
仔細觀察,動態(tài) import 經(jīng)過 webpack 編譯后,變成了下面的一坨東西:
__webpack_require__.e("src_a_js") .then(__webpack_require__.bind(__webpack_require__, "./src/a.js")) .then((res) => { console.log("動態(tài)加載a.js..", res); });
上面代碼中,__webpack_require__
用于執(zhí)行模塊,比如上面我們通過webpackChunkwebpack_demo.push
添加的模塊,里面的./src/a.js
函數(shù)就是在__webpack_require__
里面執(zhí)行的。
__webpack_require__.e
函數(shù)就是用來動態(tài)加載遠程腳本。因此,從上面的代碼中我們可以看出:
- 首先 webpack 將動態(tài) import 編譯成
__webpack_require__.e
函數(shù) __webpack_require__.e
函數(shù)加載遠程的腳本,加載完成后調(diào)用__webpack_require__
函數(shù)__webpack_require__
函數(shù)負責(zé)調(diào)用遠程腳本返回來的模塊,獲取腳本里面導(dǎo)出的對象并返回
源碼分析及實現(xiàn)
如何動態(tài)加載遠程模塊
在開始之前,我們先來看下如何使用 script 標(biāo)簽加載遠程模塊
var inProgress = {}; // url: "http://localhost:8080/src_a_js.main.js" // done: 加載完成的回調(diào) const loadScript = (url, done) => { if (inProgress[url]) { inProgress[url].push(done); return; } const script = document.createElement("script"); script.charset = "utf-8"; script.src = url; inProgress[url] = [done]; var onScriptComplete = (prev, event) => { var doneFns = inProgress[url]; delete inProgress[url]; script.parentNode && script.parentNode.removeChild(script); doneFns && doneFns.forEach((fn) => fn(event)); if (prev) return prev(event); }; script.onload = onScriptComplete.bind(null, script.onload); document.head.appendChild(script); };
loadScript(url, done)
函數(shù)比較簡單,就是通過創(chuàng)建 script 標(biāo)簽加載遠程腳本,加載完成后執(zhí)行 done 回調(diào)。inProgress
用于避免多次創(chuàng)建 script 標(biāo)簽。比如我們多次調(diào)用loadScript('http://localhost:8080/src_a_js.main.js', done)
時,應(yīng)該只創(chuàng)建一次 script 標(biāo)簽,不需要每次都創(chuàng)建。這也是為什么我們調(diào)用多次 import('a.js')
,瀏覽器 network 請求只看到家在一次腳本的原因
實際上,這就是 webpack 用于加載遠程模塊的極簡版本。
__webpack_require__.e 函數(shù)的實現(xiàn)
首先我們使用installedChunks
對象保存動態(tài)加載的模塊。key 是 chunkId
// 存儲已經(jīng)加載和正在加載的chunks,此對象存儲的是動態(tài)import的chunk,對象的key是chunkId,值為 // 以下幾種: // undefined: chunk not loaded // null: chunk preloaded/prefetched // [resolve, reject, Promise]: chunk loading // 0: chunk loaded var installedChunks = { main: 0, };
由于 import()
返回的是一個 promise,然后import()
經(jīng)過 webpack 編譯后就是一個__webpack_require__.e
函數(shù),因此可以得出__webpack_require__.e
返回的也是一個 promise,如下所示:
const scriptUrl = document.currentScript.src .replace(/#.*$/, "") .replace(/\?.*$/, "") .replace(/\/[^\/]+$/, "/"); __webpack_require__.e = (chunkId) => { return Promise.resolve(ensureChunk(chunkId, promises)); }; const ensureChunk = (chunkId) => { var installedChunkData = installedChunks[chunkId]; if (installedChunkData === 0) return; let promise; // 1.如果多次調(diào)用了__webpack_require__.e函數(shù),即多次調(diào)用import('a.js')加載相同的模塊,只要第一次的加載還沒完成,就直接使用第一次的Promise if (installedChunkData) { promise = installedChunkData[2]; } else { promise = new Promise((resolve, reject) => { // 2.注意,此時的resolve,reject還沒執(zhí)行 installedChunkData = installedChunks[chunkId] = [resolve, reject]; }); installedChunkData[2] = promise; //3. 此時的installedChunkData 為[resolve, reject, promise] var url = scriptUrl + chunkId; var error = new Error(); // 4.在script標(biāo)簽加載完成或者加載失敗后執(zhí)行l(wèi)oadingEnded方法 var loadingEnded = (event) => { if (Object.prototype.hasOwnProperty.call(installedChunks, chunkId)) { installedChunkData = installedChunks[chunkId]; if (installedChunkData !== 0) installedChunks[chunkId] = undefined; if (installedChunkData) { console.log("加載失敗....."); installedChunkData[1](error); // 5.執(zhí)行上面的reject,那resolve在哪里執(zhí)行呢? } } }; loadScript(url, loadingEnded, "chunk-" + chunkId, chunkId); } return promise; };
__webpack_require__.e
的主要邏輯在ensureChunk
方法中,注意該方法里面的第 1 到第 5 個注釋。這個方法創(chuàng)建一個 promise,并調(diào)用loadScript
方法加載動態(tài)模塊。需要特別主要的是,返回的 promise 的 resolve 方法并不是在 script 標(biāo)簽加載完成后改變。如果腳本加載錯誤或者超時,會在 loadingEnded 方法里調(diào)用 promise 的 reject 方法。
實際上,promise 的 resolve 方法是在腳本請求完成后,在 self["webpackChunkwebpack_demo"].push()執(zhí)行的時候調(diào)用的
如何執(zhí)行遠程模塊?
遠程模塊是通過self["webpackChunkwebpack_demo"].push()
函數(shù)執(zhí)行的
前面我們提到,a.js
請求返回的內(nèi)容是一個self["webpackChunkwebpack_demo"].push()
函數(shù)。當(dāng)請求完成,會自動執(zhí)行這個函數(shù)。實際上,這就是一個 jsonp 的回調(diào)方式。該方法的實現(xiàn)如下:
var webpackJsonpCallback = (data) => { var [chunkIds, moreModules] = data; var moduleId, chunkId, i = 0; for (moduleId in moreModules) { // 1.__webpack_require__.m存儲的是所有的模塊,包括靜態(tài)模塊和動態(tài)模塊 __webpack_require__.m[moduleId] = moreModules[moduleId]; } for (; i < chunkIds.length; i++) { chunkId = chunkIds[i]; if (installedChunks[chunkId]) { // 2.調(diào)用ensureChunk方法生成的promise的resolve回調(diào) installedChunks[chunkId][0](); } // 3.將該模塊標(biāo)記為0,表示已經(jīng)加載過 installedChunks[chunkId] = 0; } }; self["webpackChunkwebpack_demo"] = []; self["webpackChunkwebpack_demo"].push = webpackJsonpCallback.bind(null);
所有通過import()
加載的模塊,經(jīng)過 webpack 編譯后,都會被 self["webpackChunkwebpack_demo"].push()
包裹。
總結(jié)
在 webpack 構(gòu)建編譯階段,import()
會被編譯成類似__webpack_require__.e("src_a_js").then(__webpack_require__.bind(__webpack_require__, "./src/a.js"))
的調(diào)用方式
__webpack_require__ .e("src_a_js") .then(__webpack_require__.bind(__webpack_require__, "./src/a.js")) .then((res) => { console.log("動態(tài)加載a.js..", res); });
__webpack_require__.e()
方法會創(chuàng)建一個 script 標(biāo)簽用于請求腳本,方法執(zhí)行完返回一個 promise,此時的 promise 狀態(tài)還沒改變。
script 標(biāo)簽被添加到 document.head 后,觸發(fā)瀏覽器網(wǎng)絡(luò)請求。請求成功后,動態(tài)的腳本會自動執(zhí)行,此時self["webpackChunkwebpack_demo"].push()
方法執(zhí)行,將動態(tài)的模塊添加到__webpack_require__.m
屬性中。同時調(diào)用 promise 的 resolve 方法改變狀態(tài),模塊加載完成。
腳本執(zhí)行完成后,最后執(zhí)行 script 標(biāo)簽的 onload 回調(diào)。onload 回調(diào)主要是用于處理腳本加載失敗或者超時的場景,并調(diào)用 promise 的 reject 回調(diào),表示腳本加載失敗
以上就是Webpack學(xué)習(xí)之動態(tài)import原理及源碼分析的詳細內(nèi)容,更多關(guān)于Webpack動態(tài)import的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
JavaScript SweetAlert插件實現(xiàn)超酷消息警告框
SweetAlert是一款使用純js制作的消息警告框插件.這篇文章主要介紹了JavaScript SweetAlert插件實現(xiàn)超酷消息警告框的相關(guān)資料,需要的朋友可以參考下2016-01-01EasyUI的DataGrid綁定Json數(shù)據(jù)源的示例代碼
本篇文章主要介紹了EasyUI的DataGrid綁定Json數(shù)據(jù)源的示例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-12-12Highcharts使用簡例及異步動態(tài)讀取數(shù)據(jù)
Highcharts 是一個用純JavaScript編寫的一個圖表庫, 能夠很簡單便捷的在web網(wǎng)站或是web應(yīng)用程序添加有交互性的圖表,并且免費提供給個人學(xué)習(xí)、個人網(wǎng)站和非商業(yè)用途使用,通過本文給大家介紹Highcharts使用簡例及異步動態(tài)讀取數(shù)據(jù)的相關(guān)知識,感興趣的朋友一起學(xué)習(xí)吧2015-12-12uniapp小程序使用高德地圖api實現(xiàn)路線規(guī)劃的示例代碼
路線規(guī)劃常用于出行路線的提前預(yù)覽,我們提供4種類型的路線規(guī)劃,分別為:駕車、步行、公交和騎行,滿足各種的出行場景,這篇文章主要介紹了uniapp小程序使用高德地圖api實現(xiàn)路線規(guī)劃,需要的朋友可以參考下2023-01-01Javascript createElement和innerHTML增加頁面元素的性能對比
Javascript之createElement和innerHTML增加頁面元素的性能對比2009-09-09Javascript格式化并高亮xml字符串的方法及注意事項
這篇文章主要介紹了Javascript格式化并高亮xml字符串的方法及注意事項,需要的朋友可以參考下2018-08-08