webpack模塊化的原理解析
commonjs
在webpack中既可以書寫commonjs模塊也可以書寫es模塊,而且不用考慮瀏覽器的兼容性問題,我們來分析一下原理。
首先搞清楚commonjs模塊化的處理方式,簡單配置一下webpack,寫兩個模塊編譯一下看一下:
webpack.config.js
module.exports = { mode: "development", devtool: "none" }
index.js
const a = require('./a') console.log(a)
a.js
const a = 'a'; module.exports = a;
編譯結(jié)果
查看編譯結(jié)果,可以發(fā)現(xiàn)webpack對于每個模塊的做法類似于node,將每個模塊放在一個函數(shù)環(huán)境中并向其中傳入一些必要的參數(shù)。webpack將這些模塊組成一個對象(屬性名是模塊路徑(模塊id),屬性值為模塊內(nèi)容)傳入一個立即執(zhí)行函數(shù),立即執(zhí)行函數(shù)中定義了一個函數(shù) __webpack_require__
類似node中的require
函數(shù),實現(xiàn)了導(dǎo)入模塊的作用。
打包結(jié)果中刪去了一些注釋和暫時用不要的代碼,可以很明顯的看出來實現(xiàn)commonjs模塊化的關(guān)鍵就是這個 __webpack_require__
函數(shù),通過傳入模塊id來得到模塊的導(dǎo)出。
require 函數(shù)
__webpack_require__
函數(shù)的實現(xiàn):
function __webpack_require__(moduleId) { // Check if module is in cache if (installedModules[moduleId]) { return installedModules[moduleId].exports; } // Create a new module (and put it into the cache) var module = installedModules[moduleId] = { i: moduleId, l: false, exports: {} }; // Execute the module function modules[moduleId].call(module.exports, module, module.exports, __webpack_require__); // Flag the module as loaded module.l = true; // Return the exports of the module return module.exports; }
如果熟悉node就很容易理解這個函數(shù)了:
- 首先查看這個模塊是否已經(jīng)被加載過,所以就需要一個全局變量
installedModules
用來記錄所有被加載過模塊的導(dǎo)出 - 沒有加載過的模塊就先構(gòu)造一個
module
對象,關(guān)鍵是要有一個exports
屬性 - 執(zhí)行模塊代碼并返回模塊導(dǎo)出值
最終的一步就是需要加載啟動模塊,也就是IIFE的最后一句:
return __webpack_require__("./src/index.js");
ES Module
es 模塊化的處理方式是需要借助 __webpack_require__
實現(xiàn)的,首先看一些剛才被刪除的代碼:
__webpack_require__.r
該函數(shù)用于標(biāo)識es模塊的導(dǎo)出
// define __esModule on exports __webpack_require__.r = function (exports) { if (typeof Symbol !== 'undefined' && Symbol.toStringTag) { Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' }); } Object.defineProperty(exports, '__esModule', { value: true }); };
__webpack_require__.d
用于處理es模塊的具名導(dǎo)出
// define getter function for harmony exports __webpack_require__.d = function (exports, name, getter) { if (!__webpack_require__.o(exports, name)) { Object.defineProperty(exports, name, { enumerable: true, get: getter }); } };
__webpack_require__.o
就是給 hasOwnPreperty
換了個名字
__webpack_require__.o = function (object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
我們改一下模塊代碼看看純es Module導(dǎo)入導(dǎo)出的編譯結(jié)果:
index.js
import a, { test } from './a' import b from './b' console.log(a); test(); console.log(b)
a.js
const a = 'a'; function test() { } export default a; export { test }
b.js
const b = 'b'; export default b;
參考資料 前端進(jìn)階面試題詳細(xì)解答
編譯結(jié)果
{ "./src/a.js": (function (module, __webpack_exports__, __webpack_require__) { "use strict"; __webpack_require__.r(__webpack_exports__); /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "test", function () { return test; }); const a = 'a'; function test() { } /* harmony default export */ __webpack_exports__["default"] = (a); }), "./src/b.js": (function (module, __webpack_exports__, __webpack_require__) { "use strict"; __webpack_require__.r(__webpack_exports__); const b = 'b'; /* harmony default export */ __webpack_exports__["default"] = (b); }), "./src/index.js": (function (module, __webpack_exports__, __webpack_require__) { "use strict"; __webpack_require__.r(__webpack_exports__); /* harmony import */ var _a__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__("./src/a.js"); /* harmony import */ var _b__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__("./src/b.js"); console.log(_a__WEBPACK_IMPORTED_MODULE_0__["default"]) Object(_a__WEBPACK_IMPORTED_MODULE_0__["test"])(); console.log(_b__WEBPACK_IMPORTED_MODULE_1__["default"]) }) }
根據(jù)編譯結(jié)果可以很明白的看出來,和 commonjs 編譯出來的結(jié)果差不多,核心都是使用 __webpack_require__
函數(shù),區(qū)別在于es模塊化,exports
對象首先就會被__webpack_require__.r
標(biāo)記為es module,對于默認(rèn)導(dǎo)出就是 exports
的 default
屬性,對于具名導(dǎo)出使用 __webpack_require__.d
包裝了一下,目的是讓這些具名導(dǎo)出在模塊之外只能讀不能被修改(這是es module的特點)。
v5 的變化
但是為什么 default
沒有被__webpack_require__.d
處理,這不合理啊。本來是使用的 webpack 4打包的,然后換了webpack 5試了一下,webpack 5打包的結(jié)果中 default
也被處理了,這可能是webpack 4的一個小bug吧。
webpack5的編譯結(jié)果有些許的不同,但是整個邏輯是沒有變的:
兩種模塊化交互
webpack 是支持兩種模塊化代碼共存的,雖然不建議這樣做。首先我們先看一下他們互相導(dǎo)入的時候的導(dǎo)入結(jié)果是什么樣的:
我們來看看 webpack 是如何實現(xiàn)的,先修改一下模塊:
index.js
const { a, test } = require('./a')
a.js
import b from './b' import * as bbb from './b' console.log(bbb) console.log(b) console.log(b.b) const a = 'a'; function test() { } export default a; export { test };
b.js
module.exports = { b: () => { }, moduleName: 'b' }
編譯結(jié)果
{ "./src/a.js": (function (module, __webpack_exports__, __webpack_require__) { "use strict"; __webpack_require__.r(__webpack_exports__); /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "test", function () { return test; }); /* harmony import */ var _b__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__("./src/b.js"); /* harmony import */ var _b__WEBPACK_IMPORTED_MODULE_0___default = __webpack_require__.n(_b__WEBPACK_IMPORTED_MODULE_0__); console.log(_b__WEBPACK_IMPORTED_MODULE_0__) console.log(_b__WEBPACK_IMPORTED_MODULE_0___default.a) console.log(_b__WEBPACK_IMPORTED_MODULE_0___default.a.b) const a = 'a'; function test() { } /* harmony default export */ __webpack_exports__["default"] = (a); }), "./src/b.js": (function (module, exports) { module.exports = { b: () => { }, moduleName: 'b' } }), "./src/index.js": (function (module, exports, __webpack_require__) { const { a, test } = __webpack_require__("./src/a.js") }) }
可以發(fā)現(xiàn)當(dāng)通過es模塊的方式去 import
一個commonjs模塊時,就會把導(dǎo)入的模塊進(jìn)行一層包裝,通過 __webpack_require__.n
,主要目的應(yīng)該是為了兼容 import * as obj from '....'
這樣的語法。
該函數(shù)的具體實現(xiàn):
__webpack_require__.n = function (module) { var getter = module && module.__esModule ? function getDefault() { return module['default']; } : function getModuleExports() { return module; }; __webpack_require__.d(getter, 'a', getter); return getter; }
總結(jié)
webpack 中實現(xiàn)模塊化的核心就是 __webpack_require__
函數(shù),無論是commonjs模塊化還是es 模塊都是通過該函數(shù)來導(dǎo)入的。并且利用立即執(zhí)行函數(shù)的特點實現(xiàn)了作用域的封閉避免了全局變量的污染,非常的巧妙。
到此這篇關(guān)于webpack模塊化的原理的文章就介紹到這了,更多相關(guān)webpack模塊化內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
js實現(xiàn)一個可以兼容PC端和移動端的div拖動效果實例
這篇文章主要介紹了js實現(xiàn)一個可以兼容PC端和移動端的div拖動效果實例,具有一定的參考價值,有需要的可以了解一下。2016-12-12網(wǎng)絡(luò)之美 JavaScript中Get和Set訪問器的實現(xiàn)代碼
前兩天IE9 Beta版發(fā)布了,對于從事Web開發(fā)的朋友們來說真是個好消息啊,希望將來有一天各個瀏覽器都能遵循統(tǒng)一的標(biāo)準(zhǔn)。今天要和大家分享的是JavaScript中的Get和Set訪問器,和C#中的訪問器非常相似。2010-09-09javascript上下方向鍵控制表格行選中并高亮顯示的方法
這篇文章主要介紹了javascript上下方向鍵控制表格行選中并高亮顯示的方法,涉及javascript針對鍵盤按鍵操作相應(yīng)的技巧,具有一定參考借鑒價值,需要的朋友可以參考下2015-02-02Angular組件拿不到@Input輸入屬性問題探究解決方法
最近在工作中實現(xiàn)一個feature的時候,碰到一個小問題:Angular組件拿不到@Input輸入屬性的問題,盡管對這些問題都比較了解,但是找問題是需要一個過程的,所以還是把這個問題總結(jié)記錄了下2023-01-01JavaScript運動框架 解決速度正負(fù)取整問題(一)
這篇文章主要為大家詳細(xì)介紹了JavaScript運動框架的第一部分,解決速度正負(fù)取整問題,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-05-05js學(xué)習(xí)總結(jié)之DOM2兼容處理this問題的解決方法
這篇文章主要為大家詳細(xì)介紹了js學(xué)習(xí)總結(jié)之DOM2兼容處理this問題的解決方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-07-07