tree-shaking实战

转自  https://www.jianshu.com/p/7994b1fc6dfe

 

tree-shaking是一个在前端领域比较熟知的东西了。在没有深入了解前,一直以为他在项目中发挥了很大的作用。但是在看了许多文章说tree-shaking并没有什么卵用后,想自己深入了解一下,所以搜了许多博文,自己也在项目中试验了一下。基本了解了大致的流程。所以这篇博文主要是记录一下学习的成果。

tree-shaking是干啥的:

// app.js

export function A(a, b) {
    return a + b
}

export function B(a, b) {
    return a + b
}

// index.js

import {A} from '/app.js'

A(1, 2)

当index.js引用了app.js的A函数,如果tree-shaking起了作用,B函数是不会被打包进最后的bundle的。

但是

世界上有很多但是,而且往往但是后面的内容更加重要。

relies on the static structure of ES2015 module syntax, i.e. import and export.

在webpack官网当中有这样一句话,翻译成人话就是tree-shaking依赖es6的模块引入或输出语法。如果你的模块引入方式是require等等等乱七八糟的东西。tree-shaking将不会起到任何作用。


babel, webpack打包, uglifyJs

这三项东西东西是在我们开发中几乎绕不过去东西。而tree-shaking的关键点就在第一步,babel

虽然我不太了解webpack内部的运行机制(看过运行顺序的相关文章,但一直是懵比状态),但是看过这么多的文章后,上面三项的基本运行顺序还是理解的:

就是babel-loader先去处理js文件,处理过后,webpack进行打包处理,最后uglifyjs进行代码压缩。而关键就是babel怎么去处理js文件

babel的配置文件中有一个preset配置项:


{
  "presets": [
    ["env", {
      "modules": false  //关键点
    }],
    "stage-2",
    "react"
  ]
}


其中presets里面的env的options中有一个 modules: false,这是指示babel如何去处理import和exports等关键子,默认处理成require形式。如果加上此option,那么babel就不会吧import形式,转变成require形式。为webpack进行tree-shaking创造了条件。

在看过这些篇博文后,我本人对于tree-shaking有了一个基本的认识,那就是

babel首先处理js文件,真正进行tree-shaking识别和记录的是webpack本身。删除多于代码是在uglify中执行的

注:webpack在认定某块代码无用后,会再处理过程中写下一段注释。uglifyjs会根据这点注释去进行删除代码。

直接复制这篇博文代码

注释的大体内容(博文很久了,还是在webpack2.0时代,具体内容可能已经变化,但原理应该是不变的。)


function(module, exports, __webpack_require__) {
    /* harmony export */ exports["foo"] = foo;
    /* unused harmony export bar */;

    function foo() {
        return 'foo';
    }
    function bar() {
        return 'bar';
    }
}

tree-shaking,实战代码:

背景:在学习tree-shaking的过程中,如何支持class的tree-shaking是我一直关注的,而且大部分的文章还只停留在理论方面。所以最近自己写了一个demo,支持class的tree-shaking

1.首先使用loader去处理,实验阶段代码,编译成标准es代码。这样webpack内部的编译器才能正确识别代码。


module: {
rules: [
  {
    test: /\.js$/,
    use: [
      {
        loader: 'babel-loader',
        options: {
          presets: ['babel-preset-stage-2', 'babel-preset-react']
        }
      },
      'eslint-loader'
    ],
    exclude: /node_modules/
  }
]
}

2.然后通过webpack打包,并对代码进行tree-shaking.在打包完最后的bundle之后,和输出文件之前,对最后的bundle进行兼容性处理。

plugins: [
    new UglifyJSPlugin(),  //  uglify要在babelPugin的前面
    new BabelPlugin({  //在这个插件内部进行最后bundle的兼容性处理
      test: /\.js$/,
      babelOptions: {
        presets: [env]
      }
    }),
    new webpack.DefinePlugin({
      'process.env.NODE_ENV': JSON.stringify('production')
    }),
    new HtmlWebpackPlugin({
      template: 'index.html'
    }),
  ]


最后总结步骤:

  1. 先编译实验性质代码为标准代码,会涉及到babel-preset-stage-x插件

  2. webpack打包代码并进行tree-shaking识别。

  3. uglifyjs进行代码压缩,并根据webpack标识删除多余代码

4.对最后的代码进行兼容性处理涉及到babel-preset-env插件。

第三方类库的tree-shaking

在研究了许多第三方类库后,基本得出了一个结论:tree-shaking本质上是不能对大部分的第三方类库进行tree-shaking的.上面的实战代码,对于自己写的代码还有点用,但是只要涉及到第三方类库,基本就是歇菜。

ramda的输出文件:
大部分的react ui组件,以及函数工具类库。基本都是这样来进行模块输出,和引用的。


export { default as F } from './F';
export { default as T } from './T';
export { default as __ } from './__';
export { default as add } from './add';
export { default as addIndex } from './addIndex';
export { default as adjust } from './adjust';
export { default as all } from './all';
export { default as allPass } from './allPass';
export { default as always } from './always';
export { default as and } from './and';
export { default as any } from './any';
export { default as anyPass } from './anyPass';
export { default as ap } from './ap';
export { default as aperture } from './aperture';
export { default as append } from './append';
export { default as apply } from './apply';
export { default as applySpec } from './applySpec';

...

这样的文件结构是无法进行tree-shaking的


// 只要是你在代码中引用了一个方法,那么你肯定将所有的代码都引入了进来
import {path} from 'ramda' 


唯一的解决方法就是直接到具体的文件夹去引用,而不是在根index.js里面去引用。


import path from 'ramda/src/path'


但是如果每一次引用都是这样去写,开发的效率就无法保证,所以基本上有点追求的技术团队,基本上会再类库的基础上,开发一个babel的插件以支持代码的tree-shaking。

像著名的antd,以及ramda等都开发了相应的插件。

babel-plugin-ramda:此插件会默认将你写的代码转化为tree-shaking的代码


from:

import {path} from 'ramda' 

to

import path from 'ramda/src/path'


而本人也在了解了以上东西后,为本公司的ui组件开发了一个插件:

babel-plugin-b-rc

在看过ramda的代码,以及他的构建过程和对于tree-shaking的支持后,学到了很多。有的时候学习webpack以及如何优化网页的时候,不知道如何下手。我的经验就是找到一个技术点,进行深入。比如我得研究点就是tree-shaking,在一路研究过后,babel,babel-preset, babel-plugin,webpack,webpack-plugin,都有了一定的了解。这几个功能点基本上花费了我将近一个多月的时间。从找材料,到动手完成自己的一个babel插件,收获颇丰。学习babel以及webpack的过程,其本质是学习如何优化网页的过程!!!!



作者:strong9527
链接:https://www.jianshu.com/p/7994b1fc6dfe
来源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
posted @ 2022-02-17 18:36  hjswlqd  阅读(133)  评论(0编辑  收藏  举报