代碼執行時機將決定着是否可以正常執行,當依賴文件沒加載完成就開始執行、使用對應模塊,那麼將會致使執行異常。這在「存在資源加載失敗時,加載重試影響原來文件的執行順序」的場景下尤其常見。react
webpack 構建除了進行模塊依賴管理,實際上,也自然地管理了 entry 與 chunk 多文件的執行時機,但缺乏了對 external 文件管理,當 external 文件加載失敗或未完成時,執行、使用對應模塊一樣將致使異常。爲此,wait-external-webpack-plugin 應運而生,以 webpack 插件的形式,補充 external 的執行管理。本文將進行簡要說明。webpack
將 webpack 打包後的代碼進行簡化,其實就是一個當即調用函數;傳入「模塊」,使用 webpack_require 進行調用。在單文件下,文件加載後將當即執行業務邏輯。git
(function(modules) { // webpackBootstrap
function __webpack_require__(moduleId) {
// ...
// 執行模塊代碼
modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
}
// 引用入口
return __webpack_require__(__webpack_require__.s = "./src/entryB.js");
})({
"./entryB.js": (function(module, __webpack_exports__, __webpack_require__) {
// ...
})
});
複製代碼
爲了 「抽取公共模塊進行單獨打包避免重複加載」 或 「增長併發請求數減小總加載時間」 等緣由,通常會將代碼拆分紅多文件,可以使用以下形式:github
拆分紅多個文件後,爲了不業務邏輯執行時相關文件還沒加載完成致使執行出錯,須要等待相關文件都加載完成後再開始執行。web
entry 與 其餘 chunk 文件的 「等待-執行」 的邏輯,webpack 其實已經幫咱們自動生成了。json
// # entry.js
// 聲明依賴列表
deferredModules.push(["./src/entryA.js","commons"]);
// 緩存已完成的加載
var installedChunks = {
"entryA": 0
};
function webpackJsonpCallback(data) {
// 加載後標記完成
installedChunks[chunkId] = 0;
}
// 檢查是否都加載完成,如是,則開始執行業務邏輯
function checkDeferredModules() {
// 判斷 installedChunks 是否完整
// ...
if(fulfilled) {
// 全部都加載,開始執行
result = __webpack_require__(__webpack_require__.s = deferredModule[0]);
}
}
// 提供給 chunk 的全局回調方法
var jsonpArray = window["webpackJsonp"] = window["webpackJsonp"] || [];
jsonpArray.push = webpackJsonpCallback;
複製代碼
chunk 文件加載後,正常狀況下將調用 entry 提供的全局回調方法,標記加載完成。而當 chunk 文件先於 entry 加載完成,則會先緩存記錄,等 entry 文件加載後讀取緩存並將其標記完成。緩存
// # chunk.js
(window["webpackJsonp"] = window["webpackJsonp"] || []).push([["commons"],{
"./src/moduleA.js": (function(module, __webpack_exports__, __webpack_require__) {
// ...
})
}]);
複製代碼
基於以上分析,能夠看出 entry 和 chunk 文件加載順序不會影響執行時機,只有在都加載完成後,纔會執行業務邏輯。以下圖示前端工程師
項目引用第三方庫,通常會配置 external 讓庫單獨加載。經過 webpack 生成的代碼能夠看出,配置 external 的模塊在業務代碼執行前將被看成已存在環境中,不作任何判斷。因此當 external 文件未加載完成或加載失敗時,使用對應模塊將會致使執行出錯。併發
"react": (function(module, exports) {
eval("(function() { module.exports = window[\"React\"]; }());");
})
複製代碼
爲了不使用時出錯,在執行前需先保證 external 文件已經加載完成。處理方式以下
示意代碼:
(function () {
var entryInit = function () {
(function(modules) {
// webpackBootstrap
// ...
})({})
};
if (window["React"]) {
entryInit();
} else {
var hasInit = false;
var callback = function () {
if(hasInit) return;
if (window["React"]) {
hasInit = true;
document.removeEventListener('load', callback, true);
entryInit();
}
};
document.addEventListener('load', callback, true);
}
})();
複製代碼
等待 external 加載完成邏輯是統一的,差別在於依賴的 external 或有不一樣。爲了不手動添加出錯,咱們能夠經過以 webpack 插件的形式自動分析依賴,並生成相關代碼。
具體實現可見插件 wait-external-webpack-plugin
經過 wait-external-webpack-plugin 插件,可以自動生成等待依賴的 external 文件加載完成再執行邏輯,對開發者透明,保證文件對正常執行。
歡迎使用,歡迎任何意見或建議,謝謝。
查看更多文章 >>
github.com/joeyguo/blo…
AlloyTeam 歡迎優秀的小夥伴加入。
簡歷投遞: alloyteam@qq.com
詳情可點擊 騰訊AlloyTeam招募Web前端工程師(社招)