浅谈Webpack 是如何加载模块的


Posted in Javascript onMay 24, 2018

Webpack 在前端开发中作为模块打包工具非常受开发者的青睐,丰富的 loader 使它可以实现各种各样的功能。本文将通过 webpack 来打包一个 js 文件,看看 webpack 是如何加载各个模块的。

两个简单的源文件

为了方便分析 webpack 加载模块的原理,我们准备了两个文件:

hello.js

const hello = {
 say: arg => {
  console.info('hello ' + arg || 'world');
 }
};

export default hello;

index.js

import Hello from './hello';

Hello.say('man');

index.js 作为入口文件,引用了 hello.js 模块。

Webpack 打包

在命令行执行 webpack index.js bundle.js 对入口文件进行打包,生成 bundle.js ,大体结构为(为了方便阅读,我删除了部分多余的代码):

浅谈Webpack 是如何加载模块的

可以看到,最终生成的文件以 (function (modules) {})([模块1, 模块2]) 的方式启动,我们定义的模块被包装成一个个匿名函数,然后以数组的形式传递个一个匿名函数 function (modules) {},在这个匿名函数中定义了一个 __webpack_require__() 函数,用来加载模块,最后,通过 return __webpack_require__(__webpack_require__.s = 0); 来加载第一个模块 index.js

__webpack_require__() 函数

该函数接收一个 moduleId 作为参数,这个参数就是各个模块在数组中的索引,

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;
  /******/
 }

其中 installedModules 是用来缓存执行过的模块。通过 modules[moduleId].call() 来执行模块,最后返回模块的 exports。

模块接受的参数

以 hello.js 模块为例

(function (module, __webpack_exports__, __webpack_require__) {

  "use strict";
  const hello = {
   say: arg => {
    console.info('hello ' + arg || 'world');
   }
  };

  /* harmony default export */
  __webpack_exports__["a"] = (hello);

  /***/
 })

webpack 会向模块传递 module, __webpack_exports__, __webpack_require__ 三个参数,前两个是用来导出模块内的变量,第三个参数为前面介绍的 __webpack_require__() 的引用,用来导入其它模块。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
网页中实现浏览器的最大,最小化和关闭按钮
Mar 12 Javascript
JavaScript中实现块作用域的方法
Apr 01 Javascript
Chrome Form多次提交表单问题的解决方法
May 09 Javascript
利用js读取动态网站从服务器端返回的数据
Feb 10 Javascript
分享一则javascript 调试技巧
Jan 02 Javascript
jQuery+PHP实现动态数字展示特效
Mar 14 Javascript
基于javascript实现简单计算器功能
Jan 03 Javascript
javascript执行环境及作用域详解
May 05 Javascript
详解Vue 普通对象数据更新与 file 对象数据更新
Apr 26 Javascript
d3.js入门教程之数据绑定详解
Apr 28 Javascript
微信小程序中吸底按钮适配iPhone X方案
Nov 29 Javascript
mustache.js实现首页元件动态渲染的示例代码
Dec 28 Javascript
jquery.onoff实现简单的开关按钮功能(推荐)
May 24 #jQuery
详解javascript中的变量提升和函数提升
May 24 #Javascript
JavaScript轮播停留效果的实现思路
May 24 #Javascript
vue2单元测试环境搭建
May 24 #Javascript
Vue+mui实现图片的本地缓存示例代码
May 24 #Javascript
vue组件name的作用小结
May 23 #Javascript
linux 后台运行node服务指令方法
May 23 #Javascript
You might like
QueryPath PHP 中的jQuery
2010/04/11 PHP
Discuz!X中SESSION机制实例详解
2015/09/23 PHP
Symfony2实现在doctrine中内置数据的方法
2016/02/05 PHP
Zend Framework入门教程之Zend_Db数据库操作详解
2016/12/08 PHP
flash javascript之间的通讯方法小结
2008/12/20 Javascript
JavaScript 轻松搞定快捷留言功能 只需一行代码
2010/04/01 Javascript
window.name代替cookie的实现代码
2010/11/28 Javascript
js实现Select下拉框具有输入功能的方法
2015/02/06 Javascript
Angular8基础应用之表单及其验证
2019/08/11 Javascript
vue 框架下自定义滚动条(easyscroll)实现方法
2019/08/29 Javascript
layui之table checkbox初始化时选中对应选项的方法
2019/09/02 Javascript
[56:46]2018DOTA2亚洲邀请赛 3.31 小组赛 B组 VP vs Effect
2018/04/01 DOTA
[00:12]DAC2018 Miracle-站上中单舞台,他能否再写奇迹?
2018/04/06 DOTA
[46:58]完美世界DOTA2联赛PWL S3 Forest vs LBZS 第一场 12.17
2020/12/19 DOTA
Python批量按比例缩小图片脚本分享
2015/05/21 Python
使用Python的Scrapy框架十分钟爬取美女图
2016/12/26 Python
Python异常处理操作实例详解
2018/05/10 Python
解决pandas使用read_csv()读取文件遇到的问题
2018/06/15 Python
python实现机器学习之元线性回归
2018/09/06 Python
pyqt5移动鼠标显示坐标的方法
2019/06/21 Python
Python编译成.so文件进行加密后调用的实现
2019/12/23 Python
python如何基于redis实现ip代理池
2020/01/17 Python
使用TFRecord存取多个数据案例
2020/02/17 Python
Jupyter Notebook远程登录及密码设置操作
2020/04/10 Python
python实现扫雷小游戏
2020/04/24 Python
Python2与Python3关于字符串编码处理的差别总结
2020/09/07 Python
CSS3实例分享--超炫checkbox复选框和radio单选框
2014/09/01 HTML / CSS
Belle Maison倍美丛官网:日本千趣会旗下邮购网站
2016/07/22 全球购物
欧洲品牌瓷器餐具网上商店:Porzellantreff.de
2018/04/04 全球购物
请写出 BOOL flag 与"零值"比较的 if 语句
2016/02/29 面试题
竞争性谈判邀请书
2014/02/06 职场文书
代领报检证委托书范本
2014/10/11 职场文书
校园会短篇的广播稿
2014/10/21 职场文书
骆驼祥子读书笔记
2015/06/26 职场文书
SQL IDENTITY_INSERT作用案例详解
2021/08/23 MySQL
Java实现房屋出租系统详解
2021/10/05 Java/Android