本篇內(nèi)容主要講解“怎么理解Webpack HMR”,感興趣的朋友不妨來看看。本文介紹的方法操作簡(jiǎn)單快捷,實(shí)用性強(qiáng)。下面就讓小編來帶大家學(xué)習(xí)“怎么理解Webpack HMR”吧!
10年積累的成都網(wǎng)站制作、成都做網(wǎng)站經(jīng)驗(yàn),可以快速應(yīng)對(duì)客戶對(duì)網(wǎng)站的新想法和需求。提供各種問題對(duì)應(yīng)的解決方案。讓選擇我們的客戶得到更好、更有力的網(wǎng)絡(luò)服務(wù)。我雖然不認(rèn)識(shí)你,你也不認(rèn)識(shí)我。但先網(wǎng)站設(shè)計(jì)后付款的網(wǎng)站建設(shè)流程,更有青神免費(fèi)網(wǎng)站建設(shè)讓你可以放心的選擇與我們合作。
一、HMR 介紹
Hot Module Replacement(以下簡(jiǎn)稱:HMR 模塊熱替換)是 Webpack 提供的一個(gè)非常有用的功能,它允許在 JavaScript 運(yùn)行時(shí)更新各種模塊,而無需完全刷新。
Hot Module Replacement (or HMR) is one of the most useful features offered by webpack. It allows all kinds of modules to be updated at runtime without the need for a full refresh. --《Hot Module Replacement》
當(dāng)我們修改代碼并保存后,Webpack 將對(duì)代碼重新打包,HMR 會(huì)在應(yīng)用程序運(yùn)行過程中替換、添加或刪除模塊,而無需重新加載整個(gè)頁面。
HMR 主要通過以下幾種方式,來顯著加快開發(fā)速度:
保留在完全重新加載頁面時(shí)丟失的應(yīng)用程序狀態(tài);
只更新變更內(nèi)容,以節(jié)省寶貴的開發(fā)時(shí)間;
調(diào)整樣式更加快速 - 幾乎相當(dāng)于在瀏覽器調(diào)試器中更改樣式。
需要注意:HMR 不適用于生產(chǎn)環(huán)境,這意味著它應(yīng)當(dāng)只在開發(fā)環(huán)境使用。
二、HMR 使用方式
在 Webpack 中啟用 HMR 功能比較簡(jiǎn)單:
1. 方式一:使用 devServer
1.1 設(shè)置 devServer 選項(xiàng)
只需要在 webpack.config.js 中添加 devServer 選項(xiàng),并設(shè)置 hot 值為 true ,并使用HotModuleReplacementPlugin 和 NamedModulesPlugin (可選)兩個(gè) Plugins :
// webpack.config.js
const path = require('path')
const webpack = require('webpack')
module.exports = {
entry: './index.js',
output: {
filename: 'bundle.js',
path: path.join(__dirname, '/')
},
+ devServer: {
+ hot: true, // 啟動(dòng)模塊熱更新 HMR
+ open: true, // 開啟自動(dòng)打開瀏覽器頁面
+ },
plugins: [
+ new webpack.NamedModulesPlugin(),
+ new webpack.HotModuleReplacementPlugin()
]
}
1.2 添加 scripts
然后在 package.json 中為 scripts 命令即可:
// package.json { // ... "scripts": { + "start": "webpack-dev-server" }, // ... }
2. 方式二、使用命令行參數(shù)
另一種是通過添加 --hot 參數(shù)來實(shí)現(xiàn)。添加 --hot 參數(shù)后,devServer 會(huì)告訴 Webpack 自動(dòng)引入 HotModuleReplacementPlugin ,而不需要我們手動(dòng)引入。
另外常常也搭配 --open 來自動(dòng)打開瀏覽器到頁面。
這里移除掉前面添加的兩個(gè) Plugins :
// webpack.config.js const path = require('path') const webpack = require('webpack') module.exports = { // ... - plugins: [ - new webpack.NamedModulesPlugin(), - new webpack.HotModuleReplacementPlugin() - ] }
然后修改 package.json 文件中的 scripts 配置:
// package.json { // ... "scripts": { - "start": "webpack-dev-server" + "start": "webpack-dev-server --hot --open" }, // ... }
3. 簡(jiǎn)單示例
基于上述配置,我們簡(jiǎn)單實(shí)現(xiàn)一個(gè)場(chǎng)景:index.js 文件中導(dǎo)入 hello.js 模塊,當(dāng) hello.js 模塊發(fā)生變化時(shí), index.js 將更新模塊。
模塊代碼如下實(shí)現(xiàn):
// hello.js export default () => 'hi leo!'; // index.js import hello from './hello.js' const div = document.createElement('div'); div.innerHTML = hello(); document.body.appendChild(div);
然后在 index.html 中導(dǎo)入打包后的 JS 文件,并執(zhí)行 npm start 運(yùn)行項(xiàng)目:
了不起的 Webpack HMR 學(xué)習(xí)指南
4. 實(shí)現(xiàn)監(jiān)聽更新
當(dāng)我們通過 HotModuleReplacementPlugin 插件啟用了 HMR,則它的接口將被暴露在全局 module.hot 屬性下面。通常,可以先檢查這個(gè)接口是否可訪問,然后再開始使用它。
舉個(gè)例子,你可以這樣 accept 一個(gè)更新的模塊:
if (module.hot) { module.hot.accept('./library.js', function() { // 使用更新過的 library 模塊執(zhí)行某些操作... }) }
關(guān)于 module.hot 更多 API ,可以查看官方文檔《Hot Module Replacement API》 。
回到上面示例,我們測(cè)試更新模塊的功能。
這時(shí)我們修改 index.js 代碼,來監(jiān)聽 hello.js 模塊中的更新:
import hello from './hello.js'; const div = document.createElement('div'); div.innerHTML = hello(); document.body.appendChild(div); + if (module.hot) { + module.hot.accept('./hello.js', function() { + console.log('現(xiàn)在在更新 hello 模塊了~'); + div.innerHTML = hello(); + }) + }
然后修改 hello.js 文件內(nèi)容,測(cè)試效果:
- export default () => 'hi leo!'; + export default () => 'hi leo! hello world';
當(dāng)我們保存代碼時(shí),控制臺(tái)輸出 "現(xiàn)在在更新 hello模塊了~" ,并且頁面中 "hi leo!" 也更新為 "hi leo! hello world" ,證明我們監(jiān)聽到文件更新了。
簡(jiǎn)單 Webpack HMR 使用方式就介紹到這,更多介紹,還請(qǐng)閱讀官方文檔《Hot Module Replacement》。
5. devServer 常用配置和技巧
5.1 常用配置
根據(jù)目錄結(jié)構(gòu)的不同,contentBase、openPage 參數(shù)要配置合適的值,否則運(yùn)行時(shí)應(yīng)該不會(huì)立刻訪問到你的首頁。同時(shí)要注意你的 publicPath,靜態(tài)資源打包后生成的路徑是一個(gè)需要思考的點(diǎn),取決于你的目錄結(jié)構(gòu)。
devServer: { contentBase: path.join(__dirname, 'static'), // 告訴5.2 技巧1:文件形式輸出 dev-server 代碼
dev-server 輸出的代碼通常在內(nèi)存中,但也可以寫入硬盤,產(chǎn)出實(shí)體文件:
devServer:{ writeToDisk: true, }通常可以用于代理映射文件調(diào)試,編譯時(shí)會(huì)產(chǎn)出許多帶 hash 的 js 文件,不帶 hash 的文件同樣也是實(shí)時(shí)編譯的。
5.3 技巧2:默認(rèn)使用本地 IP 啟動(dòng)服務(wù)
有的時(shí)候,啟動(dòng)服務(wù)時(shí),想要默認(rèn)使用本地的 ip 地址打開:
devServer:{ disableHostCheck: true, // true:不進(jìn)行host檢查 // useLocalIp: true, // 建議不在這里配置 // host: '0.0.0.0', // 建議不在這里配置 }同時(shí)還需要將 host 配置為 0.0.0.0,這個(gè)配置建議在 scripts 命令中追加,而非在配置中寫死,否則將來不想要這種方式往回改折騰,取巧一點(diǎn),配個(gè)新命令:
"dev-ip": "yarn run dev --host 0.0.0.0 --useLocalIp"5.4 技巧3:指定啟動(dòng)的調(diào)試域名
有時(shí)啟動(dòng)的時(shí)候希望是指定的調(diào)試域名,例如:local.test.baidu.com:
devServer:{ open: true, public: 'local.test.baidu.com:8080', // 需要帶上端口 port: 8080, }同時(shí)需要將 127.0.0.1 修改為指定的 host,可以借助 iHost 等工具去修改,各個(gè)工具大同小異,格式如下:
127.0.0.1 local.test.baidu.com服務(wù)啟動(dòng)后將自動(dòng)打開 local.test.baidu.com:8080 訪問
5.5 技巧4:?jiǎn)?dòng) gzip 壓縮
devServer:{ compress: true, }三、HMR 基本原理介紹
從前面介紹中,我們知道:HMR 主要功能是會(huì)在應(yīng)用程序運(yùn)行過程中替換、添加或刪除模塊,而無需重新加載整個(gè)頁面。
那么,Webpack 編譯源碼所產(chǎn)生的文件變化在編譯時(shí),替換模塊實(shí)現(xiàn)在運(yùn)行時(shí),兩者如何聯(lián)系起來?
帶著這兩個(gè)問題,我們先簡(jiǎn)單看下 HMR 核心工作流程(簡(jiǎn)化版):
HMR 工作流程圖.png
接下來開始 HMR 工作流程分析:
當(dāng) Webpack(Watchman) 監(jiān)聽到項(xiàng)目中的文件/模塊代碼發(fā)生變化后,將變化通知 Webpack 中的構(gòu)建工具(Packager)即 HMR Plugin;
然后經(jīng)過 HMR Plugin 處理后,將結(jié)果發(fā)送到應(yīng)用程序(Application)的運(yùn)行時(shí)框架(HMR Runtime);
最后由 HMR Runtime 將這些發(fā)生變化的文件/模塊更新(新增/刪除或替換)到模塊系統(tǒng)中。
其中,HMR Runtime 是構(gòu)建工具在編譯時(shí)注入的,通過統(tǒng)一的 Module ID 將編譯時(shí)的文件與運(yùn)行時(shí)的模塊對(duì)應(yīng)起來,并且對(duì)外提供一系列 API 供應(yīng)用層框架(如 React)調(diào)用。
注意?:建議先理解上面這張圖的大致流程,在進(jìn)行后續(xù)閱讀。放心,我等著大家~?
四、HMR 完整原理和源碼分析
通過上一節(jié)內(nèi)容,我們大概知道 HMR 簡(jiǎn)單工作流程,那么或許你現(xiàn)在可能還有很多疑惑:文件更新是什么通知 HMR Plugin?HMR Plugin 怎么發(fā)送更新到 HMR Runtime?等等問題。
那么接下來我們開始詳細(xì)結(jié)合源碼分析整個(gè) HMR 模塊熱更新流程,首先還是先看流程圖,可以先不了解圖中方法名稱(紅色字體黃色背景色部分):
Webpack HMR.png
上圖展示了從我們修改代碼,到模塊熱更新完成的一個(gè) HMR 完整工作流程,圖中已用紅色阿拉伯?dāng)?shù)字符號(hào)將流程標(biāo)識(shí)出來。
要了解上面工作原理,我們先理解圖中這幾個(gè)名稱概念:
Webpack-dev-server :一個(gè)服務(wù)器插件,相當(dāng)于 express 服務(wù)器,啟動(dòng)一個(gè) Web 服務(wù),只適用于開發(fā)環(huán)境;
Webpack-dev-middleware :一個(gè) Webpack-dev-server 的中間件,作用簡(jiǎn)單總結(jié)為:通過watch mode,監(jiān)聽資源的變更,然后自動(dòng)打包。
Webpack-hot-middleware :結(jié)合 Webpack-dev-middleware 使用的中間件,它可以實(shí)現(xiàn)瀏覽器的無刷新更新,也就是 HMR;
下面一起學(xué)習(xí) HMR 整個(gè)工作原理吧:
1.監(jiān)控代碼變化,重新編譯打包
首先根據(jù) devServer 配置,使用 npm start 將啟動(dòng) Webpack-dev-server 啟動(dòng)本地服務(wù)器并進(jìn)入 Webpack 的 watch 模式,然后初始化 Webpack-dev-middleware ,在 Webpack-dev-middleware 中通過調(diào)用 startWatch() 方法對(duì)文件系統(tǒng)進(jìn)行 watch:
// webpack-dev-server\bin\webpack-dev-server.js // 1.啟動(dòng)本地服務(wù)器 Line 386 server = new Server(compiler, options); // webpack-dev-server\lib\Server.js // 2.初始化 Webpack-dev-middleware Line 109 this.middleware = webpackDevMiddleware(compiler, Object.assign({}, options, wdmOptions)); // webpack-dev-middleware\lib\Shared.js // 3.開始 watch 文件系統(tǒng) Line 171 startWatch: function() { //... // start watching if(!options.lazy) { var watching = compiler.watch(options.watchOptions, share.handleCompilerCallback); context.watching = watching; } //... } share.startWatch(); // ...當(dāng) startWatch() 方法執(zhí)行后,便進(jìn)入 watch 模式,若發(fā)現(xiàn)文件中代碼發(fā)生修改,則根據(jù)配置文件對(duì)模塊重新編譯打包。
2.保存編譯結(jié)果
Webpack 與 Webpack-dev-middleware 交互,Webpack-dev-middleware 調(diào)用 Webpack 的 API 對(duì)代碼變化進(jìn)行監(jiān)控,并通知 Webpack 將重新編譯的代碼通過 JavaScript 對(duì)象保存在內(nèi)存中。
我們會(huì)發(fā)現(xiàn),在 output.path 指定的 dist 目錄并沒有保存編譯結(jié)果的文件,這是為什么?
其實(shí), Webpack 將編譯結(jié)果保存在內(nèi)存中,因?yàn)樵L問內(nèi)存中的代碼比訪問文件系統(tǒng)中的文件快,這樣可以減少代碼寫入文件的開銷。
Webpack 能將代碼保存到內(nèi)存中,需要?dú)w功于 Webpack-dev-middleware 的 memory-fs 依賴庫,它將原本 outputFileSystem 替換成了 MemoryFileSystem 的實(shí)例,便實(shí)現(xiàn)代碼輸出到內(nèi)存中。其中部分源碼如下:
// webpack-dev-middleware\lib\Shared.js Line 108 // store our files in memory var fs; var isMemoryFs = !compiler.compilers && compiler.outputFileSystem instanceof MemoryFileSystem; if(isMemoryFs) { fs = compiler.outputFileSystem; } else { fs = compiler.outputFileSystem = new MemoryFileSystem(); } context.fs = fs;上述代碼先判斷 fileSystem 是否是 MemoryFileSystem 的實(shí)例,若不是,則用 MemoryFileSystem 的實(shí)例替換 compiler 之前的 outputFileSystem。這樣 bundle.js 文件代碼就作為一個(gè)簡(jiǎn)單 JavaScript 對(duì)象保存在內(nèi)存中,當(dāng)瀏覽器請(qǐng)求 bundle.js 文件時(shí),devServer 就直接去內(nèi)存中找到上面保存的 JavaScript 對(duì)象并返回給瀏覽器端。
3.監(jiān)控文件變化,刷新瀏覽器
Webpack-dev-server 開始監(jiān)控文件變化,與第 1 步不同的是,這里并不是監(jiān)控代碼變化重新編譯打包。
當(dāng)我們?cè)谂渲梦募信渲昧?devServer.watchContentBase 為 true ,Webpack-dev-server 會(huì)監(jiān)聽配置文件夾中靜態(tài)文件的變化,發(fā)生變化時(shí),通知瀏覽器端對(duì)應(yīng)用進(jìn)行瀏覽器刷新,這與 HMR 不一樣。
// webpack-dev-server\lib\Server.js // 1. 讀取參數(shù) Line 385 if (options.watchContentBase) { defaultFeatures.push('watchContentBase'); } // 2. 定義 _watch 方法 Line 697 Server.prototype._watch = function (watchPath) { // ... const watcher = chokidar.watch(watchPath, options).on('change', () => { this.sockWrite(this.sockets, 'content-changed'); }); this.contentBaseWatchers.push(watcher); }; // 3. 執(zhí)行 _watch() 監(jiān)聽文件變化 Line 339 watchContentBase: () => { if (/^(https?:)?\/\//.test(contentBase) || typeof contentBase === 'number') { throw new Error('Watching remote files is not supported.'); } else if (Array.isArray(contentBase)) { contentBase.forEach((item) => { this._watch(item); }); } else { this._watch(contentBase); } }4.建立 WS,同步編譯階段狀態(tài)
這一步都是 Webpack-dev-server 中處理,主要通過 sockjs(Webpack-dev-server 的依賴),在 Webpack-dev-server 的瀏覽器端(Client)和服務(wù)器端(Webpack-dev-middleware)之間建立 WebSocket 長連接。
然后將 Webpack 編譯打包的各個(gè)階段狀態(tài)信息同步到瀏覽器端。其中有兩個(gè)重要步驟:
發(fā)送狀態(tài)
Webpack-dev-server 通過 Webpack API 監(jiān)聽 compile 的 done 事件,當(dāng) compile 完成后,Webpack-dev-server 通過 _sendStats 方法將編譯后新模塊的 hash 值用 socket 發(fā)送給瀏覽器端。
保存狀態(tài)
瀏覽器端將_sendStats 發(fā)送過來的 hash 保存下來,它將會(huì)用到后模塊熱更新。
// webpack-dev-server\lib\Server.js // 1. 定義 _sendStats 方法 Line 685 // send stats to a socket or multiple sockets Server.prototype._sendStats = function (sockets, stats, force) { //... this.sockWrite(sockets, 'hash', stats.hash); }; // 2. 監(jiān)聽 done 事件 Line 86 compiler.plugin('done', (stats) => { // 將最新打包文件的 hash 值(stats.hash)作為參數(shù)傳入 _sendStats() this._sendStats(this.sockets, stats.toJson(clientStats)); this._stats = stats; }); // webpack-dev-server\client\index.js // 3. 保存 hash 值 Line 74 var onSocketMsg = { // ... hash: function hash(_hash) { currentHash = _hash; }, // ... } socket(socketUrl, onSocketMsg);5.瀏覽器端發(fā)布消息
當(dāng) hash 消息發(fā)送完成后,socket 還會(huì)發(fā)送一條 ok 的消息告知 Webpack-dev-server,由于客戶端(Client)并不請(qǐng)求熱更新代碼,也不執(zhí)行熱更新模塊操作,因此通過 emit 一個(gè) "webpackHotUpdate" 消息,將工作轉(zhuǎn)交回 Webpack。
// webpack-dev-server\client\index.js // 1. 處理 ok 消息 Line 135 var onSocketMsg = { // ... ok: function ok() { sendMsg('Ok'); if (useWarningOverlay || useErrorOverlay) overlay.clear(); if (initial) return initial = false; // eslint-disable-line no-return-assign reloadApp(); }, // ... } // 2. 處理刷新 APP Line 218 function reloadApp() { // ... if (_hot) { // 動(dòng)態(tài)加載 emitter var hotEmitter = require('webpack/hot/emitter'); hotEmitter.emit('webpackHotUpdate', currentHash); if (typeof self !== 'undefined' && self.window) { // broadcast update to window self.postMessage('webpackHotUpdate' + currentHash, '*'); } } // ... }6.傳遞 hash 到 HMR
Webpack/hot/dev-server 監(jiān)聽瀏覽器端 webpackHotUpdate 消息,將新模塊 hash 值傳到客戶端 HMR 核心中樞的 HotModuleReplacement.runtime ,并調(diào)用 check 方法檢測(cè)更新,判斷是瀏覽器刷新還是模塊熱更新。如果是瀏覽器刷新的話,則沒有后續(xù)步驟咯~~
// webpack\hot\dev-server.js // 1.監(jiān)聽 webpackHotUpdate Line 42 var hotEmitter = require("./emitter"); hotEmitter.on("webpackHotUpdate", function(currentHash) { lastHash = currentHash; if(!upToDate() && module.hot.status() === "idle") { log("info", "[HMR] Checking for updates on the server..."); check(); } }); var check = function check() { module.hot.check(true).then(function(updatedModules) { if(!updatedModules) { // ... window.location.reload();// 瀏覽器刷新 return; } if(!upToDate()) { check(); } }).catch(function(err) { /*...*/}); }; // webpack\lib\HotModuleReplacement.runtime.js // 3.調(diào)用 HotModuleReplacement.runtime 定義的 check 方法 Line 167 function hotCheck(apply) { if(hotStatus !== "idle") throw new Error("check() is only allowed in idle status"); hotApplyOnUpdate = apply; hotSetStatus("check"); return hotDownloadManifest(hotRequestTimeout).then(function(update) { //... }); }7.檢測(cè)是否存在更新
當(dāng) HotModuleReplacement.runtime 調(diào)用 check 方法時(shí),會(huì)調(diào)用 JsonpMainTemplate.runtime 中的 hotDownloadUpdateChunk (獲取最新模塊代碼)和 hotDownloadManifest (獲取是否有更新文件)兩個(gè)方法,這兩個(gè)方法的源碼,在下一步展開。
// webpack\lib\HotModuleReplacement.runtime.js // 1.調(diào)用 HotModuleReplacement.runtime 定義 hotDownloadUpdateChunk 方法 Line 171 function hotCheck(apply) { if(hotStatus !== "idle") throw new Error("check() is only allowed in idle status"); hotApplyOnUpdate = apply; hotSetStatus("check"); return hotDownloadManifest(hotRequestTimeout).then(function(update) { //... { // hotEnsureUpdateChunk 方法中會(huì)調(diào)用 hotDownloadUpdateChunk hotEnsureUpdateChunk(chunkId); } }); }其中 hotEnsureUpdateChunk 方法中會(huì)調(diào)用 hotDownloadUpdateChunk :
// webpack\lib\HotModuleReplacement.runtime.js Line 215 function hotEnsureUpdateChunk(chunkId) { if(!hotAvailableFilesMap[chunkId]) { hotWaitingFilesMap[chunkId] = true; } else { hotRequestedFilesMap[chunkId] = true; hotWaitingFiles++; hotDownloadUpdateChunk(chunkId); } }8.請(qǐng)求更新最新文件列表
在調(diào)用 check 方法時(shí),會(huì)先調(diào)用 JsonpMainTemplate.runtime 中的 hotDownloadManifest 方法, 通過向服務(wù)端發(fā)起 AJAX 請(qǐng)求獲取是否有更新文件,如果有的話將 mainfest 返回給瀏覽器端。
這邊涉及一些原生 XMLHttpRequest,就不全部貼出了~
// webpack\lib\JsonpMainTemplate.runtime.js // hotDownloadManifest 定義 Line 22 function hotDownloadManifest(requestTimeout) { return new Promise(function(resolve, reject) { try { var request = new XMLHttpRequest(); var requestPath = $require$.p + $hotMainFilename$; request.open("GET", requestPath, true); request.timeout = requestTimeout; request.send(null); } catch(err) { return reject(err); } request.onreadystatechange = function() { // ... }; }); }9.請(qǐng)求更新最新模塊代碼
在 hotDownloadManifest 方法中,還會(huì)執(zhí)行 hotDownloadUpdateChunk 方法,通過 JSONP 請(qǐng)求最新的模塊代碼,并將代碼返回給 HMR runtime 。
然后 HMR runtime 會(huì)將新代碼進(jìn)一步處理,判斷是瀏覽器刷新還是模塊熱更新。
// webpack\lib\JsonpMainTemplate.runtime.js // hotDownloadManifest 定義 Line 12 function hotDownloadUpdateChunk(chunkId) { // 創(chuàng)建 script 標(biāo)簽,發(fā)起 JSONP 請(qǐng)求 var head = document.getElementsByTagName("head")[0]; var script = document.createElement("script"); script.type = "text/javascript"; script.charset = "utf-8"; script.src = $require$.p + $hotChunkFilename$; $crossOriginLoading$; head.appendChild(script); }10.更新模塊和依賴引用
這一步是整個(gè)模塊熱更新(HMR)的核心步驟,通過 HMR runtime 的 hotApply 方法,移除過期模塊和代碼,并添加新的模塊和代碼實(shí)現(xiàn)熱更新。
從 hotApply 方法可以看出,模塊熱替換主要分三個(gè)階段:
鴻蒙官方戰(zhàn)略合作共建——HarmonyOS技術(shù)社區(qū)
找出過期模塊 outdatedModules 和過期依賴 outdatedDependencies ;
// webpack\lib\HotModuleReplacement.runtime.js // 找出 outdatedModules 和 outdatedDependencies Line 342 function hotApply() { // ... var outdatedDependencies = {}; var outdatedModules = []; function getAffectedStuff(updateModuleId) { var outdatedModules = [updateModuleId]; var outdatedDependencies = {}; // ... return { type: "accepted", moduleId: updateModuleId, outdatedModules: outdatedModules, outdatedDependencies: outdatedDependencies }; }; function addAllToSet(a, b) { for (var i = 0; i < b.length; i++) { var item = b[i]; if (a.indexOf(item) < 0) a.push(item); } } for(var id in hotUpdate) { if(Object.prototype.hasOwnProperty.call(hotUpdate, id)) { // ... 省略多余代碼 if(hotUpdate[id]) { result = getAffectedStuff(moduleId); } if(doApply) { for(moduleId in result.outdatedDependencies) { // 添加到 outdatedDependencies addAllToSet(outdatedDependencies[moduleId], result.outdatedDependencies[moduleId]); } } if(doDispose) { // 添加到 outdatedModules addAllToSet(outdatedModules, [result.moduleId]); appliedUpdate[moduleId] = warnUnexpectedRequire; } } } }2. 從緩存中刪除過期模塊、依賴和所有子元素的引用;
// webpack\lib\HotModuleReplacement.runtime.js // 從緩存中刪除過期模塊、依賴和所有子元素的引用 Line 442 function hotApply() { // ... var idx; var queue = outdatedModules.slice(); while(queue.length > 0) { moduleId = queue.pop(); module = installedModules[moduleId]; // ... // 移除緩存中的模塊 delete installedModules[moduleId]; // 移除過期依賴中不需要使用的處理方法 delete outdatedDependencies[moduleId]; // 移除所有子元素的引用 for(j = 0; j < module.children.length; j++) { var child = installedModules[module.children[j]]; if(!child) continue; idx = child.parents.indexOf(moduleId); if(idx >= 0) { child.parents.splice(idx, 1); } } } // 從模塊子組件中刪除過時(shí)的依賴項(xiàng) var dependency; var moduleOutdatedDependencies; for(moduleId in outdatedDependencies) { if(Object.prototype.hasOwnProperty.call(outdatedDependencies, moduleId)) { module = installedModules[moduleId]; if(module) { moduleOutdatedDependencies = outdatedDependencies[moduleId]; for(j = 0; j < moduleOutdatedDependencies.length; j++) { dependency = moduleOutdatedDependencies[j]; idx = module.children.indexOf(dependency); if(idx >= 0) module.children.splice(idx, 1); } } } } }3. 將新模塊代碼添加到 modules 中,當(dāng)下次調(diào)用 __webpack_require__ (webpack 重寫的 require 方法)方法的時(shí)候,就是獲取到了新的模塊代碼了。
// webpack\lib\HotModuleReplacement.runtime.js // 將新模塊代碼添加到 modules 中 Line 501 function hotApply() { // ... for(moduleId in appliedUpdate) { if(Object.prototype.hasOwnProperty.call(appliedUpdate, moduleId)) { modules[moduleId] = appliedUpdate[moduleId]; } } }hotApply 方法執(zhí)行之后,新代碼已經(jīng)替換舊代碼,但是我們業(yè)務(wù)代碼并不知道這些變化,因此需要通過 accept事件通知應(yīng)用層使用新的模塊進(jìn)行“局部刷新”,我們?cè)跇I(yè)務(wù)中是這么使用:
if (module.hot) { module.hot.accept('./library.js', function() { // 使用更新過的 library 模塊執(zhí)行某些操作... }) }11.熱更新錯(cuò)誤處理
在熱更新過程中,hotApply 過程中可能出現(xiàn) abort 或者 fail 錯(cuò)誤,則熱更新退回到刷新瀏覽器(Browser Reload),整個(gè)模塊熱更新完成。
// webpack\hot\dev-server.js Line 13 module.hot.check(true).then(function (updatedModules) { if (!updatedModules) { return window.location.reload(); } // ... }).catch(function (err) { var status = module.hot.status(); if (["abort", "fail"].indexOf(status) >= 0) { window.location.reload(); } });到此,相信大家對(duì)“怎么理解Webpack HMR”有了更深的了解,不妨來實(shí)際操作一番吧!這里是創(chuàng)新互聯(lián)網(wǎng)站,更多相關(guān)內(nèi)容可以進(jìn)入相關(guān)頻道進(jìn)行查詢,關(guān)注我們,繼續(xù)學(xué)習(xí)!
新聞名稱:怎么理解WebpackHMR
本文URL:http://weahome.cn/article/peosjd.html