# webpack构建产物分析
我们在使用
webpack
构建项目后,会发现打包出来的代码非常的“丑”,这是为什么?原因就是:webpack
支持多种模块规范,但是最后都会变成commonJS
规范(webpack5
对纯esm
做了一定的优化),但是浏览器不支持commonJS
规范,于是webpack
自己实现了require
和module.exports
,所以会有很多polyfill
代码的注入。
# 简单分析
// webpack相关依赖版本
// "webpack": "^5.76.3",
// "webpack-cli": "^5.0.1",
// "webpack-dev-server": "^4.13.1"
// commonjs 中加载 commonjs
(() => {
var __webpack_modules__ = ({
/***/ "./src/add-content.js":
/*!****************************!*\
!*** ./src/add-content.js ***!
\****************************/
/***/ (() => {
eval("modules.exports = {\n write: function() {\n document.write('i am using a config file')\n }\n}\n\n//# sourceURL=webpack://webpack-basic-build/./src/add-content.js?");
/***/ }),
"./src/index.js": (__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => {
eval("const { write } = __webpack_require__(/*! ./add-content */ \"./src/add-content.js\")\n\nwrite()\n\n//# sourceURL=webpack://webpack-basic-build/./src/index.js?");
}
})
var __webpack_module_cache__ = {};
/******/ // The require function
/******/ function __webpack_require__(moduleId) {
/******/ // Check if module is in cache
/******/ var cachedModule = __webpack_module_cache__[moduleId];
/******/ if (cachedModule !== undefined) {
/******/ return cachedModule.exports;
/******/ }
/******/ // Create a new module (and put it into the cache)
/******/ var module = __webpack_module_cache__[moduleId] = {
/******/ // no module.id needed
/******/ // no module.loaded needed
/******/ exports: {}
/******/ };
/******/
/******/ // Execute the module function
/******/ __webpack_modules__[moduleId](module, module.exports, __webpack_require__);
/******/
/******/ // Return the exports of the module
/******/ return module.exports;
/******/ }
var __webpack_exports__ = __webpack_require__("./src/index.js");
})()
// es6 module中加载es6module
/******/ (() => { // webpackBootstrap
/******/ "use strict";
/******/ var __webpack_modules__ = ({
/***/ "./src/add-content.js":
/*!****************************!*\
!*** ./src/add-content.js ***!
\****************************/
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"default\": () => (/* export default binding */ __WEBPACK_DEFAULT_EXPORT__)\n/* harmony export */ });\n/* harmony default export */ function __WEBPACK_DEFAULT_EXPORT__() {\n // es6 模块化\n document.write('i am using a config file')\n}\n\n//# sourceURL=webpack://webpack-basic-build/./src/add-content.js?");
/***/ }),
/***/ "./src/index.js":
/*!**********************!*\
!*** ./src/index.js ***!
\**********************/
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var _add_content__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./add-content */ \"./src/add-content.js\");\n\n\n\n(0,_add_content__WEBPACK_IMPORTED_MODULE_0__[\"default\"])()\n\n//# sourceURL=webpack://webpack-basic-build/./src/index.js?");
/***/ })
/******/ });
/************************************************************************/
/******/ // The module cache
/******/ var __webpack_module_cache__ = {};
/******/
/******/ // The require function
/******/ function __webpack_require__(moduleId) {
/******/ // Check if module is in cache
/******/ var cachedModule = __webpack_module_cache__[moduleId];
/******/ if (cachedModule !== undefined) {
/******/ return cachedModule.exports;
/******/ }
/******/ // Create a new module (and put it into the cache)
/******/ var module = __webpack_module_cache__[moduleId] = {
/******/ // no module.id needed
/******/ // no module.loaded needed
/******/ exports: {}
/******/ };
/******/
/******/ // Execute the module function
/******/ __webpack_modules__[moduleId](module, module.exports, __webpack_require__);
/******/
/******/ // Return the exports of the module
/******/ return module.exports;
/******/ }
/******/
/************************************************************************/
/******/ /* webpack/runtime/define property getters */
/******/ (() => {
/******/ // define getter functions for harmony exports
/******/ __webpack_require__.d = (exports, definition) => {
/******/ for(var key in definition) {
/******/ if(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {
/******/ Object.defineProperty(exports, key, { enumerable: true, get: definition[key] });
/******/ }
/******/ }
/******/ };
/******/ })();
/******/
/******/ /* webpack/runtime/hasOwnProperty shorthand */
/******/ (() => {
/******/ __webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))
/******/ })();
/******/
/******/ /* webpack/runtime/make namespace object */
/******/ (() => {
/******/ // define __esModule on exports
/******/ __webpack_require__.r = (exports) => {
/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
/******/ }
/******/ Object.defineProperty(exports, '__esModule', { value: true });
/******/ };
/******/ })();
/******/
/************************************************************************/
/******/
/******/ // startup
/******/ // Load entry module and return exports
/******/ // This entry module can't be inlined because the eval devtool is used.
/******/ var __webpack_exports__ = __webpack_require__("./src/index.js");
/******/
/******/ })()
;
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
- 最外层匿名函数。它用来包裹整个
bundle
,并构成自身的作用域。 __webpack_module_cache__
对象。每个模块只在第一次被加载的时候执行,之后其导出值就被存储到这个对象里面,当再次被加载的时候webpack
会直接从这里取值,而不会重新执行该模块。__webpack_require__
函数。对模块加载的实现,在浏览器中可以通过调用__webpack_require__(module_id)
来完成模块导入。webpack
打包后的代码能够运行, 是因为webpack
根据commonJS
规范实现了一个require
方法__webpack_modules__
对象。工程中所有产生了依赖关系的模块都会以key-value
的形式放在这里。key
可以理解为一个模块的id
,由数字或者一个很短的hash
字符串构成;value
则是由一个匿名函数包裹的模块实体,匿名函数的参数赋予了每个模块导出和导入的能力。
bundle
是如何在浏览器中执行的
- 在最外层匿名函数中初始化浏览器执行环境,包括定义
__webpack_module_cache__
对象、__webpack_require__
函数等,为模块的加载和执行做一些准备工作。 - 加载入口模块。每个
bundle
都有且只有一个入口模块,在上面的代码中,index.js
是入口模块,在浏览器中会从它开始执行。 - 执行模块代码。如果执行到了
module.exports
则记录下模块的导出值;如果中间遇到require
函数(准确地说是__webpack_require__
),则会暂时交出执行权,进入__webpack_require__
函数体内进行加载其他模块的逻辑。 - 在
__webpack_require__
中判断即将加载的模块是否存在于__webpack_module_cache__
中。如果存在则直接取值,否则回到第3步,执行该模块的代码来获取导出值。 - 所有依赖的模块都已执行完毕,最后执行权又回到入口模块。当入口模块的代码执行完毕,也就意味着整个
bundle
运行结束。
TIP
第 3
步和第 4
步是一个递归的过程。Webpack
为每个模块创造了一个可以导出和导入模块的环境,但本质上并没有修改代码的执行逻辑,因此代码执行的顺序与模块加载的顺序是完全一致的,这也是 Webpack
模块打包的奥秘。