在上周末广州举办的feday中,webpack的核心开发者Sean在介绍webpack插件系统原理时,隆重介绍了一个中国学生于Google夏令营,在导师Tobias带领下写的一个webpack插件,webpack-deep-scope-analysis-plugin,这个插件可以大大提升webpack tree-shaking的效率。前端
tree-shaking 做为 rollup 的一个杀手级特性,可以利用ES6的静态引入规范,减小包的体积,避免没必要要的代码引入,webpack2也很快引入了这个特性,可是目前,webpack只能作比较简单的解决方案,好比:webpack
import { isNumber, isNull } from 'lodash-es' export function fun1() { // do something } export function isNull(...args) { return isNull(...args) }
这个例子中,webpack会寻找引入变量的引用,当发现没有对isNumber的引用时,就会去除isNumber的代码。这其实不太实用,毕竟在如今的vscode中,没有引用的变量在ide中都会灰显提示,通常不会犯这种import某个模块却不用的错误了。git
若是是接下来这种引入方式呢,我写了一个demo以下github
这个例子很是简单,若是用图来表示是这样web
在index.js中引入了func.js中的func2,并无引入func1,可是func1引入了lodash。webpack检查的时候发现func.js中的确用到了lodash,因此不会把lodash去掉。实际上,咱们根本没用到它。json
webpack-deep-scope-analysis-plugin就能够解决这种判断。babel
引入前ide
引入后函数
85.8kb -> 不到1kb 工具
固然,我这里是标题党了,由于这里直接把一个lodash库给去掉了,因此变化才这么惊人。可是即便在实际项目中,咱们也能轻易用一个插件减小大量的没必要要的引入。
原理方面,能够参考做者的Medium文章,解释得很是清楚,这里不作赘述。
webpack的原理,其实就是遍历全部的模块,把它们打包成一个文件,在这个过程当中,它就知道哪些export的模块有被使用到。那咱们一样也能够遍历全部的scope(做用域),简化没有用到的scope,最后只留下咱们须要的。
咱们提到的这个webpack插件,正是内置了这样一个scope分析器,它可以从入口文件中分析出scope的引用关系,最后排除掉全部没有用到的模块。
固然,这个插件也并非本身作了全部的事情,它也是依赖于了前人的工做。 escope 是一个分析ES中scope的工具,插件做者将它改为了ts版本集成到了插件中,而且利用了webpack暴露的接口,能够解析出来的模块的AST树,基于这个AST就能够交给escope分析出scope的引用关系。
一些使用此插件的特殊状况也能够参考做者的原Medium文章。
首先,要用到tree-shaking,必然要保证引用的模块都是ES6规范的。这也是为何我在前面的demo中,引入的是lodash-es
而不是lodash
。
在项目中,注意要把babel
设置module: false
,避免babel将模块转为CommonJS规范。引入的模块包,也必须是符合ES6规范,而且在最新的webpack中加了一条限制,即在package.json
中定义sideEffect: false
,这也是为了不出现import xxx
致使模块内部的一些函数执行后影响全局环境,却被去除掉的状况。
当时跟这位插件做者沟通,他说未来有可能Tobias会把这个插件内置到webpack中,这也是符合webpack4零配置的趋势。可是咱们也看获得,要将前端工程的dead code elimination作到和其余静态语言同样好,靠这些工具是远远不够的,模块自身也必须配合作到符合规范。
github项目地址:https://github.com/vincentdch...
Medium博客地址:https://medium.com/webpack/be...