webpack
做为前端最火的构建工具,是前端自动化工具链最重要的部分,使用门槛较高。本系列是笔者本身的学习记录,比较基础,但愿经过问题 + 解决方式的模式,之前端构建中遇到的具体需求为出发点,学习webpack
工具中相应的处理办法。(本篇中的参数配置及使用方式均基于webpack4.0版本
)javascript
使用webpack
对脚本进行合并是很是方便的,由于webpack
实现了对各类不一样模块规范的兼容处理,对前端开发者来讲,理解这种实现方式比学习如何配置webpack
更为重要,本节的内容实用性较低。
html
脚本合并是基于模块化规范的,javascript
模块化是一个很是混乱的话题,各类**【*MD】**规范乱飞还要外加一堆【*.js】的规范实现。现代化前端项目多基于框架进行开发,较为流行的框架内部基本已经统一遵循ES6
的模块化标准,尽管支持度不一,但经过构建工具能够解决浏览器支持滞后的问题;基于nodejs
的服务端项目原生支持CommonJs
标准;而开发中引入的一些工具类的库,热门的工具类库为了能同时兼容浏览器和node环境,一般会使用UMD
标准(Universal Module Definition) 来实现模块化,对UMD
范式不了解的读者能够先阅读《javascript基础修炼(4)——UMD规范的代码推演》一文,甚至有些第三方库并无遵循任何模块化方案。若是不借助构建工具,想要对各种方案实现兼容是很是复杂的。前端
webpack
默认支持的是CommonJs
规范,毕竟它是nodejs
支持的模块管理方式,而没有node
哪来的webpack
。但同时为了扩展其使用场景,webpack
在版本迭代中也加入了对ES harmony
规范和AMD
规范的兼容。java
webpack
打包后输出文件的基本结构是下面这个样子的:node
(function(modules) { // webpackBootstrap // 模块缓存对象 var installedModules = {}; // webpack内部的模块引用函数 function __webpack_require__(moduleId) { // 加载入口JS // 输出 return module.exports; } // 挂载模块数组 __webpack_require__.m = modules; // ... // 在__webpack_require__挂载多个属性 // 传入入口JS模块ID执行函数并输出模块 return __webpack_require__(__webpack_require__.s = 0); }); // 包含全部模块的数组 ([ /* id为0 */ (function(module, exports) { console.log('1') }) ]);
简化之后实际上就是一个自执行函数:webpack
(function(modules){ return __webpack_require__(0); }([Module0,Module1...]))
能够看到__webpack_reqruie__( )
这个方法的参数就是模块的惟一ID标识,返回值就是module.exports
,因此webpack
对于CommonJs
规范是原生支持的。web
对于ES Harmony
规范不熟悉的能够查看《ES6 Module语法》一文。express
先使用import
命令加载一个CommonJs
规范导出的模块,查看打包后的代码能够看到模块引用的部分被转换成了下面这样:数组
__webpack_require__.r(__webpack_exports__); /* harmony import */ var _components_component10k_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__("./components/component10k.js"); /* harmony import */ var _components_component10k_js__WEBPACK_IMPORTED_MODULE_0___default = __webpack_require__.n(_components_component10k_js__WEBPACK_IMPORTED_MODULE_0__);
简化一下再来看:浏览器
__webpack_require__.r(__webpack_exports__); var a = __webpack_require__("./components/component10k.js"); var b = __webpack_require__.n(a);
这里涉及到两个工具函数:
这个方法是给模块的exports
对象加上ES Harmony
规范的标记,若是支持Symbol
对象,则为exports
对象的Symbol.toStringTag属性赋值Module,这样作的结果是exports
对象在调用toString方法时会返回'Module'(笔者并无查到这种写法的原因);若是不支持Symbol
对象,则将exports.__esModule
赋值为true。
另外一个工具函数是:
传入了一个模块,返回一个getter方法,此处是一个高阶函数的应用,实现的功能是当模块的__esModule
属性为真时,返回一个getDefault( )
方法,不然返回getModuleExports( )
方法.
回过头再来看上面的简化代码:
// 添加ES Harmony规范模块标记 __webpack_require__.r(__webpack_exports__); // a实际上获得了模块经过module.exports输出的对象 var a = __webpack_require__("./components/component10k.js"); // 根据a的模块化规范类型返回不一样的getter函数,当getter函数执行时才会真正获得模块对象 var b = __webpack_require__.n(a);
总结一下,
webpack
所作的处理至关于对模块增长了代理,若是被加载模块符合ES Harmony
规范,则返回module['default']
,不然返回module
。这里的module泛指模块输出的对象。
再使用import
加载一个使用export
语法输出的ES Harmony
模块,查看打包结果中的模块文件能够看到:
//component10k.js模块文件在main.bundle.js中的内容 __webpack_require__.r(__webpack_exports__); __webpack_exports__["default"] = (function(){ Array.from('component10k'); })
能够看到输出的内容直接绑定到了输出模块的default属性上,因为这个模块被打上了__esModule
的标记,因此引用它的模块会经过module['default']来取用其内容,也就正好命中了模块的输出内容。
咱们将component10k.js
模块改成用AMD
规范定义:
define(function(){ console.log('test'); })
查看通过webpack
打包后,这个模块变成了以下的样子:
var __WEBPACK_AMD_DEFINE_RESULT__; !(__WEBPACK_AMD_DEFINE_RESULT__ = (function(){ console.log('test'); }).call(exports, __webpack_require__, exports, module), __WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__));
简化一下:
var result; !(result=(function(){}).call(...),result!==undefined && module.exports = result);
抽象一下:
var result; !(expression1,expression2 && expression3)
这里涉及的javascript的基本知识较多,逗号表达式的优先级最低,因此最后参与运算,逗号表达式会从左到右依次执行语句,并返回最后一个表达式的结果,&&为短路运算语法,即前一个条件成立时才计算后面的表达式,赋值语句执行完后会将所赋的值返回。此处外层的!(expression )
语法起了什么做用,笔者也没看懂,但愿了解的读者多多指教。
因此,
webpack
对于AMD
模块的处理,其实是加了一层封装,将模块运行的结果挂载到了webpack
模块的module.exports对象上。