webpack 核心原理

1. 怎样才能运行 import / export

  • 不同浏览器功能不同
    现代浏览器可以通过 <script type=module> 来支持 import export
    IE 8~15 不支持 import export,所以不可能运行
  • 兼容策略
    1). 激进的兼容策略:把代码全放在 <script type=module> 里
    缺点:不被 IE 8~15 支持;而且会导致文件请求过多。
    index.html
<body>
<script type="module" src="index.js"></script>

</body>

运行 http-server project_1/

它会把所有的文件都请求一遍

2). 平稳的兼容策略:把关键字转译为普通代码,并把所有文件打包成一个文件
缺点:需要写复杂的代码来完成这件事情

2. 把关键字转移成普通代码

将 import / export 转成函数

使用 @babel/core ,在上一节的 deps_4.ts 里添加下面几行代码

import * as babel from '@babel/core';
 const code = readFileSync(filepath).toString()
+ const { code: es5Code } = babel.transform(code, {
+    presets: ['@babel/preset-env']
+  })
  // 初始化 depRelation[key]
+  depRelation[key] = { deps: [], code: es5Code }

运行 node -r ts-node/register bundler_1.ts

a.js 的变化
1). import 关键字 -> 变成了 require()
2). export 关键字 -> 变成 exports['default']

伏笔
这里的 code 是字符串

a.js 变成 ES5 之后的代码详解

疑惑1

Object.defineProperty(exports, "__esModule", {value: true});
这是在做啥?

  • 解惑
    给当前模块添加 __esModule: true 属性,方便跟 CommonJS 模块区分开
    那为什么不直接用 exports.__esModule = true;
    两种区别不大,上面写法功能更强,exports.__esModule 兼容性更好
疑惑2

exports["default"] = void 0;
这是在做啥?
解惑
void 0 等价于 undefined,老 JSer 的常见过时技巧
这句话是为了强制清空 exports['default'] 的值

细节1

import b from './b.js' 变成了
var _b = _interopRequireDefault(require("./b.js"))
b.value 变成了
_b['default'].value
解释: _interopRequireDefault(module)

_ 下划线前缀是为了避免与其他变量重名
该函数的意图是给模块添加 'default'

为什么要加 default?
CommonJS 模块没有默认导出,加上方便兼容

内部实现:return m && m.__esModule ? m : { "default": m }
其他 _interop 开头的函数大多都是为了兼容旧代码

细节2

export default a 变成了
var _default = a; exports["default"] = _default;
简化一下就是 exports["default"] = a

const x = 'x'; export {x} 会变成 var x = 'x'; exports.x = x
解释
这个 _default 中间变量有什么意义我也没看出来,也许后面有用
其他部分都挺好理解的

import 关键字会变成 require 函数�
export 关键字会变成 exports 对象

  • 本质:ESModule 语法变成了 CommonJS 规则

3. 把所有的文件打包成一个

  • 打包成一个什么样的文件?
    包含了所有模块,然后能执行所有模块
    比如:
var depRelation = [ 
  {key: 'index.js', deps: ['a.js', 'b.js'], code: function... },
  {key: 'a.js', deps: ['b.js'], code: function... },
  {key: 'b.js', deps: ['a.js'], code: function... }
] // 为什么把 depRelation 从对象改为数组?
// 因为数组的第一项就是入口,而对象没有第一项的概念
execute(depRelation[0].key) // 执行入口文件
function execute(key){
  var item = depRelation.find(i => i.key === key)
  item.code(???) // 执行 item 的代码,因此 code 最好是个函数,方便执行
  // 但是目前还不知道要传什么参数给 code 
  // 代码待完善
}

现在有三个问题还没解决
1). depRelation 是对象,需要变成一个数组
2). code 是字符串,需要变成一个函数
3). execute 函数待完善

3.1 把 depRelation 改为一个数组

复制 bundle_1.ts 修改如下代码

type DepRelation = { key: string,  deps: string[], code: string }[];
// 初始化一个空的 depRelation,用于收集依赖
const depRelation: DepRelation = [];
const item = { deps: [], code: es5Code }

traverse(ast, {
    enter: path => {
      if (path.node.type === 'ImportDeclaration') {
        item.deps.push(depProjectPath)
      }
    }
  })

3.2 把 code 由字符串改为函数

上面代码 code2 加上${code2}后就是一个函数了
require, module, exports 这三个参数是 CommonJS 2 规范规定的

3.3 完善 execute 函数(主体思路)

const modules = {} // modules 用于缓存所有模块�function execute(key) { 
  if (modules[key]) { return modules[key] }
  var item = depRelation.find(i => i.key === key)
  var require = (path) => {
    return execute(pathToKey(path)) // 把相对路径变成 key 比如./b.js => b.js
  }
  modules[key] = { __esModule: true } // modules['a.js'] 给 a.js 准备一个空对象方便它去挂载
  var module = { exports: modules[key] }
  item.code(require, module, module.exports)  // 执行 a.js 的代码执行后就会挂载到 module.exports 上面
  return module.exports
}

3.4 最终文件主要内容

var depRelation = [ 
  {key: 'index.js', deps: ['a.js', 'b.js'], code: function... },
  {key: 'a.js', deps: ['b.js'], code: function... },
  {key: 'b.js', deps: ['a.js'], code: function... }
] 
var modules = {} // modules 用于缓存所有模块
execute(depRelation[0].key)
function execute(key){
  var require = ...
  var module = ...
  item.code(require, module, module.exports)
  ...
}
// 详见 dist.js

dist.js 代码
https://github.com/wanglifa/webapck-demo-2/blob/main/dist.js

虽然我们已经知道了最终文件的主要内容,但是怎么才能得到这个最终文件那?
答:拼凑出字符串,然后写入文件

var dist = ""; 
dist += content; 
writeFileSync('dist.js', dist)

3.5 自动创建最终文件

  • bundler_3.ts(基于 bundler_2.ts 复制修改的)
+ import { writeFileSync } from 'fs'
+ writeFileSync('dist_2.js', generateCode())

+ function generateCode() {
  let code = ''
  code += 'var depRelation = [' + depRelation.map(item => {
    const { key, deps, code } = item
    return `{
      key: ${JSON.stringify(key)}, 
      deps: ${JSON.stringify(deps)},
      code: function(require, module, exports){
        ${code}
      }
    }`
  }).join(',') + '];\n'
  code += 'var modules = {};\n'
  code += `execute(depRelation[0].key)\n`
  code += `
  function execute(key) {
    if (modules[key]) { return modules[key] }
    var item = depRelation.find(i => i.key === key)
    if (!item) { throw new Error(\`\${item} is not found\`) }
    var pathToKey = (path) => {
      var dirname = key.substring(0, key.lastIndexOf('/') + 1)
      var projectPath = (dirname + path).replace(\/\\.\\\/\/g, '').replace(\/\\\/\\\/\/, '/')
      return projectPath
    }
    var require = (path) => {
      return execute(pathToKey(path))
    }
    modules[key] = { __esModule: true }
    var module = { exports: modules[key] }
    item.code(require, module, module.exports)
    return modules[key]
  }
  `
  return code
+ }

运行 node -r ts-node/register bundler_3.ts
得到新文件 dist_2.js,与 dist.js 相差无几

3.6 目前还存在的问题

问题列表
1). 生成的代码中有多个重复的 _interopXXX 函数
2). 只能引入和运行 JS 文件
3). 只能理解 import,无法理解 require
4). 不支持插件
5). 不支持配置入口文件和 dist 文件名

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,524评论 5 460
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,869评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,813评论 0 320
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,210评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,085评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,117评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,533评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,219评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,487评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,582评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,362评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,218评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,589评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,899评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,176评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,503评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,707评论 2 335

推荐阅读更多精彩内容