官网地址:https://rollupjs.org/guide/en/
我们先看看 Rollup 的作者 Rich Harris 是怎么讲的?Rollup 是一个模块化的打包工具。本质上,它会合并 JavaScript 文件。而且你不需要去手动指定它们的顺序,或者去担心文件之间的变量名冲突。它的内部实现会比说的复杂一点,但是它就是这么做的 —— 合并。
webpack 对前端来说是再熟悉不过的工具了,它提供了强大的功能来构建前端的资源,包括 html/js/ts/css/less/scss ... 等语言脚本,也包括 images/fonts ... 等二进制文件。正是因为 webpack 拥有如此强大的功能,所以 webpack 在进行资源打包的时候,就会产生很多冗余的代码(如果你有查看过 webpack 的 bundle 文件,便会发现)。
而对于一些项目(特别是类库)只有 js,而没有其他的静态资源文件,使用 webpack 就有点大才小用了,因为 webpack bundle 文件的体积略大,运行略慢,可读性略低。这个时候就可以选择 Rollup
Rollup是一个模块打包器,支持 ES6 模块,支持 Tree-shaking,但不支持 webpack 的 code-splitting、模块热更新等,这意味着它更适合用来做类库项目的打包器而不是应用程序项目的打包器。
对于应用适用于 webpack,对于类库更适用于 Rollup,react/vue/anngular 都在用Rollup作为打包工具
在阐述 Rollup 的构建原理之前,我们需要了解一些前置知识
magic-string 是 Rollup 作者写的一个关于字符串操作的库,这个库主要是对字符串一些常用方法进行了封装
- var MagicString = require('magic-string')
- var magicString = new MagicString('export var name = "zhangsan"')
- // 以下所有操作都是基于原生字符串
- // 类似于截取字符串
- console.log(magicString.snip(0, 6).toString()) // export
- // 从开始到结束删除
- console.log(magicString.remove(0, 7).toString()) // var name = "zhangsan"
- // 多个模块,把他们打包在一个文件里,需要把很多文件的源代码合并在一起
- let bundleString = new MagicString.Bundle();
- bundleString.addSource({
- content: 'console.log(hello)',
- separator: '\n'
- })
- bundleString.addSource({
- content: 'console.log(world)',
- separator: '\n'
- })
- // // 原理类似
- // let str = ''
- // str += 'console.log(hello);\n'
- // str += 'console.log(world);\n'
- console.log(bundleString.toString())
- // hello
- // world
通过 javascript parse 可以把代码转化为一颗抽象语法树 AST,这颗树定义了代码的结构,通过操纵这个树,我们可以精确的定位到声明语句、赋值语句、运算符语句等等,实现对代码的分析、优化、变更等操作 源代码:main.js
- // main.js
- import { a } from './a'
- console.log(a)
转化为 AST 是长这样子的,如下:
- {
- "type": "Program", // 这个 AST 类型为 Program,表明是一个程序
- "start": 0,
- "end": 40,
- "body": [ // body 是一个数组,每一条语句都对应 body 下的一个语句
- {
- "type": "ImportDeclaration", // 导入声明类型
- "start": 0,
- "end": 23,
- "specifiers": [
- {
- "type": "ImportSpecifier",
- "start": 9,
- "end": 10,
- "imported": {
- "type": "Identifier",
- "start": 9,
- "end": 10,
- "name": "a" // 导入模块命名 name 'a'
- },
- "local": {
- "type": "Identifier",
- "start": 9,
- "end": 10,
- "name": "a" // 本地模块命名,同 imported.name
- }
- }
- ],
- "source": {
- "type": "Literal",
- "start": 18,
- "end": 23,
- "value": "./a", // 导入路径 './a'
- "raw": "'./a'"
- }
- },
- {
- "type": "ExpressionStatement", // 表达式类型
- "start": 24,
- "end": 38,
- "expression": {
- "type": "CallExpression", // 调用表达式类型
- "start": 24,
- "end": 38,
- "callee": {
- "type": "MemberExpression",
- "start": 24,
- "end": 35,
- "object": {
- "type": "Identifier",
- "start": 24,
- "end": 31,
- "name": "console"
- },
- "property": {
- "type": "Identifier",
- "start": 32,
- "end": 35,
- "name": "log"
- },
- "computed": false,
- "optional": false
- },
- "arguments": [
- {
- "type": "Identifier",
- "start": 36,
- "end": 37,
- "name": "a"
- }
- ],
- "optional": false
- }
- }
- ],
- "sourceType": "module"
- }
Parse(解析)将代码转化成抽象语法树,树上有很多的 estree 节点 Transform(转换) 对抽象语法树进行转换 Generate(代码生成) 将上一步经过转换过的抽象语法树生成新的代码
acorn 是一个 JavaScript 语法解析器,它将 JavaScript 字符串解析成语法抽象树 AST 如果想了解 AST 语法树可以点下这个网址https://astexplorer.net/
在 js 中,作用域是用来规定变量访问范围的规则, 作用域链是由当前执行环境和上层执行环境的一系列变量对象组成的,它保证了当前执行环境对符合访问权限的变量和函数的有序访问
你给它一个入口文件 —— 通常是 index.js。Rollup 将使用 Acorn 读取解析文件 —— 将返回给我们一种叫抽象语法树(AST)的东西。一旦有了 AST ,你就可以发现许多关于代码的东西,比如它包含哪些 import 声明。
假设 index.js 文件头部有这样一行:
- import foo from './foo.js';
这就意味着 Rollup 需要去加载,解析,分析在 index.js 中引入的 ./foo.js。重复解析直到没有更多的模块被加载进来。更重要的是,所有的这些操作都是可插拔的,所以您可以从 node_modules 中导入或者使用 sourcemap-aware 的方式将 ES2015 编译成 ES5 代码。
在 Rollup 中,一个文件就是一个模块,每个模块都会根据文件的代码生成一个 AST 抽象语法树。
分析 AST 节点,就是看这个节点有没有调用函数方法,有没有读到变量,有,就查看是否在当前作用域,如果不在就往上找,直到找到模块顶级作用域为止。如果本模块都没找到,说明这个函数、方法依赖于其他模块,需要从其他模块引入。如果发现其他模块中有方法依赖其他模块,就会递归读取其他模块,如此循环直到没有依赖的模块为止 找到这些变量或着方法是在哪里定义的,把定义语句包含进来即可 其他无关代码一律不要
看如下代码,我们先实际操作一下:
- // index.js
- import { foo } from "./foo";
- foo()
- var city = 'hangzhou'
- function test() {
- console.log('test')
- }
- console.log(test())
- // foo.js
- import { bar } from "./bar";
- export function foo() {
- console.log('foo')
- }
- // bar.js
- export function bar() {
- console.log('bar')
- }
- // rollup.config.js
- export default {
- input: './src/index.js',
- output: {
- file: './dist/bundle.js', // 打包后的存放文件
- format: 'cjs', //输出格式 amd es6 life umd cjs
- name: 'bundleName', //如果输出格式 life,umd 需要指定一个全局变量
- }
- };
执行 npm run build,会得到如下结果:
- 'use strict';
- function foo() {
- console.log('foo');
- }
- foo();
- function test() {
- console.log('test');
- }
- console.log(test());
以上,我们可以看到Rollup 只是会合并你的代码 —— 没有任何浪费。所产生的包也可以更好的缩小。有人称之为 “作用域提升(scope hoisting)”。其次,它把你导入的模块中的未使用代码移除。这被称为“(摇树优化)treeshaking”。总之,Rollup 就是一个模块化的打包工具。
接下来我们进入源码,具体分析下 Rollup 的构建流程
- │ bundle.js // Bundle 打包器,在打包过程中会生成一个 bundle 实例,用于收集其他模块的代码,最后再将收集的代码打包到一起。
- │ external-module.js // ExternalModule 外部模块,例如引入了 'path' 模块,就会生成一个 ExternalModule 实例。
- │ module.js // Module 模块,module 实例。
- │ rollup.js // rollup 函数,一切的开始,调用它进行打包。
- │
- ├─ast // ast 目录,包含了和 AST 相关的类和函数
- │ analyse.js // 主要用于分析 AST 节点的作用域和依赖项。
- │ Scope.js // 在分析 AST 节点时为每一个节点生成对应的 Scope 实例,主要是记录每个 AST 节点对应的作用域。
- │ walk.js // walk 就是递归调用 AST 节点进行分析。
- │
- ├─finalisers
- │ cjs.js
- │ index.js
- │
- └─utils // 一些帮助函数
- map-helpers.js
- object.js
- promise.js
- replaceIdentifiers.js
我们以 index.js 入口文件,index 依赖了 foo.js,foo 依赖了 bar.js
- // index.js
- import { foo } from "./foo";
- foo()
- var city = 'hangzhou'
- function test() {
- console.log('test')
- }
- console.log(test())
- // foo.js
- import { bar } from "./bar";
- export function foo() {
- console.log('foo')
- }
- // bar.js
- export function bar() {
- console.log('bar')
- }
- // debug.js
- const path = require('path')
- const rollup = require('./lib/rollup')
- // 入口文件的绝对路径
- let entry = path.resolve(__dirname, 'src/main.js')
- // 和源码有所不同,这里使用的是同步,增加可读性
- rollup(entry, 'bundle.js')
首先生成一个 Bundle 实例,也就是打包器。然后执行 build 打包编译
- // rollup.js
- let Bundle = require('./bundle')
- function rollup(entry, outputFileName) {
- // Bundle 代表打包对象,里面包含所有的模块信息
- const bundle = new Bundle({ entry })
- // 调用 build 方法开始进行编译
- bundle.build(outputFileName)
- }
- module.exports = rollup
lib/bundle.js根据入口路径出发(在 bundle 中,我们会首先统一处理下入口文件的后缀),去找到他的模块定义,在 fetchModule 中,会生成一个 module 实例
我们关注红框中的代码,会发现返回了一个 module
每个文件都是一个模块,每个模块都会有一个 Module 实例。在 Module 实例中,会调用 acorn 库的 parse() 方法将代码解析成 AST。
对生成的 AST 进行分析analyse我们先看一下入口文件 index.js 生成的 AST
可以看到 ast.body 是一个数组,分别对应 index.js 的五条语句 展开这个 ast 树如下:
- {
- "type": "Program",
- "start": 0,
- "end": 128,
- "body": [
- {
- "type": "ImportDeclaration", // 导入声明
- "start": 0,
- "end": 31,
- "specifiers": [
- {
- "type": "ImportSpecifier",
- "start": 9,
- "end": 12,
- "imported": {
- "type": "Identifier",
- "start": 9,
- "end": 12,
- "name": "foo"
- },
- "local": {
- "type": "Identifier",
- "start": 9,
- "end": 12,
- "name": "foo"
- }
- }
- ],
- "source": {
- "type": "Literal",
- "start": 20,
- "end": 30,
- "value": "./foo.js",
- "raw": "\"./foo.js\""
- }
- },
- {
- "type": "ExpressionStatement",
- "start": 32,
- "end": 37,
- "expression": {
- "type": "CallExpression",
- "start": 32,
- "end": 37,
- "callee": {
- "type": "Identifier",
- "start": 32,
- "end": 35,
- "name": "foo"
- },
- "arguments": [],
- "optional": false
- }
- },
- {
- "type": "VariableDeclaration",
- "start": 38,
- "end": 59,
- "declarations": [
- {
- "type": "VariableDeclarator",
- "start": 42,
- "end": 59,
- "id": {
- "type": "Identifier",
- "start": 42,
- "end": 46,
- "name": "city"
- },
- "init": {
- "type": "Literal",
- "start": 49,
- "end": 59,
- "value": "hangzhou",
- "raw": "'hangzhou'"
- }
- }
- ],
- "kind": "var"
- },
- {
- "type": "FunctionDeclaration",
- "start": 61,
- "end": 104,
- "id": {
- "type": "Identifier",
- "start": 70,
- "end": 74,
- "name": "test"
- },
- "expression": false,
- "generator": false,
- "async": false,
- "params": [],
- "body": {
- "type": "BlockStatement",
- "start": 77,
- "end": 104,
- "body": [
- {
- "type": "ExpressionStatement",
- "start": 83,
- "end": 102,
- "expression": {
- "type": "CallExpression",
- "start": 83,
- "end": 102,
- "callee": {
- "type": "MemberExpression",
- "start": 83,
- "end": 94,
- "object": {
- "type": "Identifier",
- "start": 83,
- "end": 90,
- "name": "console"
- },
- "property": {
- "type": "Identifier",
- "start": 91,
- "end": 94,
- "name": "log"
- },
- "computed": false,
- "optional": false
- },
- "arguments": [
- {
- "type": "Literal",
- "start": 95,
- "end": 101,
- "value": "test",
- "raw": "'test'"
- }
- ],
- "optional": false
- }
- }
- ]
- }
- },
- {
- "type": "ExpressionStatement",
- "start": 106,
- "end": 125,
- "expression": {
- "type": "CallExpression",
- "start": 106,
- "end": 125,
- "callee": {
- "type": "MemberExpression",
- "start": 106,
- "end": 117,
- "object": {
- "type": "Identifier",
- "start": 106,
- "end": 113,
- "name": "console"
- },
- "property": {
- "type": "Identifier",
- "start": 114,
- "end": 117,
- "name": "log"
- },
- "computed": false,
- "optional": false
- },
- "arguments": [
- {
- "type": "CallExpression",
- "start": 118,
- "end": 124,
- "callee": {
- "type": "Identifier",
- "start": 118,
- "end": 122,
- "name": "test"
- },
- "arguments": [],
- "optional": false
- }
- ],
- "optional": false
- }
- }
- ],
- "sourceType": "module"
- }
我们通过这个 AST 树,分析 **analyse **具体做了什么???
第一步:分析当前模块导入【import】和导出【exports】模块,将引入的模块和导出的模块存储起来this.imports = {};//存放着当前模块所有的导入
this.exports = {};//存放着当前模块所有的导出
- this.imports = {};//存放着当前模块所有的导入
- this.exports = {};//存放着当前模块所有的导出
- this.ast.body.forEach(node => {
- if (node.type === 'ImportDeclaration') {// 说明这是一个 import 语句
- let source = node.source.value; // 从哪个模块导入的
- let specifiers = node.specifiers; // 导入标识符
- specifiers.forEach(specifier => {
- const name = specifier.imported.name; //name
- const localName = specifier.local.name; //name
- //本地的哪个变量,是从哪个模块的的哪个变量导出的
- this.imports[localName] = { name, localName, source }
- });
- //}else if(/^Export/.test(node.type)){ // 导出方法有很多
- } else if (node.type === 'ExportNamedDeclaration') { // 说明这是一个 exports 语句
- let declaration = node.declaration;//VariableDeclaration
- if (declaration.type === 'VariableDeclaration') {
- let name = declaration.declarations[0].id.name;
- this.exports[name] = {
- node, localName: name, expression: declaration
- }
- }
- }
- });
- analyse(this.ast, this.code, this);//找到了_defines 和 _dependsOn
打断点可以看到,foo 已经被存入 imports =》** import { foo } from "./foo"; ** exports:{} 表示没有导出语句
第二步:analyse(this.ast, this.code, this); //找到_defines 和 _dependsOn
找出当前模块使用到了哪些变量 标记哪些变量时当前模块声明的,哪些变量是导入别的模块的变量 我们定义以下字段用来存放:_defines: { value: {} },//存放当前模块定义的所有的全局变量
_dependsOn: { value: {} },//当前模块没有定义但是使用到的变量,也就是依赖的外部变量_included: { value: false, writable: true },//此语句是否已经被包含到打包结果中,防止重复打包_source: { value: magicString.snip(statement.start, statement.end) } //magicString.snip 返回的还是 magicString 实例 clone
分析每个 AST 节点之间的作用域,构建 scope tree,
- function analyse(ast, magicString, module) {
- let scope = new Scope();//先创建一个模块内的全局作用域
- //遍历当前的所有的语法树的所有的顶级节点
- ast.body.forEach(statement => {
- //给作用域添加变量 var function const let 变量声明
- function addToScope(declaration) {
- var name = declaration.id.name;//获得这个声明的变量
- scope.add(name);
- if (!scope.parent) {//如果当前是全局作用域的话
- statement._defines[name] = true;
- }
- }
- Object.defineProperties(statement, {
- _defines: { value: {} },//存放当前模块定义的所有的全局变量
- _dependsOn: { value: {} },//当前模块没有定义但是使用到的变量,也就是依赖的外部变量
- _included: { value: false, writable: true },//此语句是否已经 被包含到打包结果中了
- //start 指的是此节点在源代码中的起始索引,end 就是结束索引
- //magicString.snip 返回的还是 magicString 实例 clone
- _source: { value: magicString.snip(statement.start, statement.end) }
- });
- //这一步在构建我们的作用域链
- walk(statement, {
- enter(node) {
- let newScope;
- &nb
名称栏目:Rollup-构建原理及简易实现
网页网址:http://www.csdahua.cn/qtweb/news27/441427.html网站建设、网络推广公司-快上网,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等
声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 快上网