Tree Shaking

这几天整理的一下过往的文章和笔记,备份到了 Github 上,地址👉 blog

如果我的内容帮助到了您,欢迎点个 Star 🎉🎉🎉 鼓励鼓励 :) ~~ 👆


浏览器原生 ES 模块和性能问题

ES 模块为您提供了一种原生方式,可以将代码分解为更小的模块化部分,并将变量和函数的范围限制在需要的地方。

当您 import 导入函数或变量时,必须下载该模块的整个文件。如果只从一个包含数百个函数的文件中导入一个函数,那么最终下载的 JavaScript 将远远超过实际需要的。

例如,假设我们有一个导出三个实用函数的工具库。

// utils.js
export function shuffle() {}
export function foo() {}
export function baz() {}

export { shuffle, foo, baz }

在另一个文件中,我们要使用 shuffle() 函数,所以我们使用 import 导入它。

import { shuffle } from './path/to/utils.js'

let name = ['O.O', 'D.O', 'K.O', 'O.K']
shuffle(name)

如果您使用浏览器原生 ES 模块,使用 [type="module"] 加载脚本,则整个 utils.js 文件将由浏览器下载、编译和解析。

您只需要 shuffle(),但浏览器必须抓取整个文件才能为您获取该函数。可以想象,这是一个性能问题,尤其是对于较大的库。

这就是需要 Tree Shaking 的原因。

什么是 Tree Shaking,它是如何工作的?

模块打包器是获取 JavaScript 文件并将所有 import 函数和变量组合或连接到单个文件中的工具。

为了尽可能提高文件的性能,现代的模块打包器,比如 webpackrollup.jsesbuild 使用一个名为 tree shaking 的过程。

Tree Shaking 指基于 ES Module 进行静态分析,通过 AST 将用不到的函数进行移除,从而减小打包体积。

当他们在文件中遇到 import 时,他们只会导入您指定的函数和变量(以及他们使用的任何内部变量或函数)。使用上面的示例,捆绑文件将仅包含 shuffle() 函数和 name 数组。

function shuffle (arr) {
  // do something...
}

let name = ['Gandalf', 'Radagast', 'Merlin']
shuffle(name)

浏览器原生 ES 模块还有其他性能挑战,因此如果您正在使用它们,我建议您使用模块打包器。

webpack 的 Tree Shaking

从 webpack 4+ 开始,生产环境下 webpack 会自动为您优化和减少捆绑包的大小,它使用的优化技术是 Tree Shaking。本质上,它是一种用于删除未使用代码的优化技术,通过分析静态的 ES 模块,来移除未使用的代码。

您还可以通过在配置文件中添加带有特定字段的 optimization 对象来手动优化应用程序。

minimize 字段用于指示 webpack 最小化包大小。默认情况下,webpack 将尝试使用 TerserPlugin 来实现这一点,它是 webpack 附带的一个代码压缩包。

minimize 适用于通过从代码中删除不必要的数据来最小化代码,从而减少处理后生成的代码大小。

我们还可以通过在 optimization 对象中添加一个 minimizer 数组字段来使用其他首选缩小器。下面的 uglifyjs-webpack-plugin 就是一个例子。

// webpack.config.js
const Uglify = require('uglifyjs-webpack-plugin')

module.exports = {
  optimization {
    minimize : true,
    minimizer : [
      new Uglify({
        cache : true,
        test: /\.js(\?.*)?$/i,
      })
    ]
  } 
}

uglifyjs-webpack-plugin 有两个非常重要的选项。首先,启用 cache 意味着只有当现有文件发生新更改时,Uglify 才会缩小它们,而 test 选项指定了我们要缩小的特定文件类型。

注意:uglifyjs-webpack-plugin 提供了一个全面的选项列表,可用于缩小代码。

引入支持 Tree Shaking 的 Package

为了减小生产环境体积,我们可以使用一些支持 ES 的 package,比如使用 lodash-es 替代 lodash

我们可以在 npm.devtool中查看某个库是否支持 Tree Shaking。

查看两者对比:

lodash-es
lodash

更多资料

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

推荐阅读更多精彩内容