本节内容派生于以下链接指向的内容 ,并遵守 CC BY 4.0 许可证的规定。
以下内容如果没有特殊声明,可以认为都是基于原内容的修改和删减后的结果。
Rspack 支持 tree shaking 功能,这是一个在 JavaScript 生态中广泛使用的术语,主要用于去除未被访问的代码,俗称“死代码”。当一个模块的某些导出未被使用且不存在副作用时,这部分代码就可以被安全地删除,以减小最终产物的体积。
你可以将应用程序想象成一棵树。绿色表示实际用到的源码和库,是树上活的树叶。灰色表示未引用代码,是秋天树上枯萎的树叶。为了除去死去的树叶,你必须摇动(shake)这棵树,使它们落下。
需要注意的是,Rspack 本身不直接删除死代码,而是标记未使用的导出为潜在的"死代码"。这些代码能被后续的压缩工具识别并处理。因此,如果 压缩功能 被关闭,你将不会看到代码有任何实际的删除效果。
死代码(dead code) 是指程序中一段已经不会被执行的代码,通常是因为重构、优化或者逻辑错误导致的。这些代码可能是之前版本的遗留物,或者某些条件下永远不会被执行的代码。
为了有效利用 tree shaking 功能,你需要:
production
以启用相关优化。
mode
默认为 production
。import
和 export
)。
modules
选项设置为 false
。当 mode 被设置为 production
时,Rspack 会默认启用一系列与 tree shaking 相关的优化措施,包括:
下面有一些例子来说明各个配置项的作用,为了增强文档的可读性,我们会使用伪代码来展示代码的删除效果。
让我们通过一个例子来更好地理解这一机制,假设 src/main.js
是项目的入口文件:
在上述示例中,util.js
中的 bar
没有被使用。在 production
模式下,Rspack 会默认启用 usedExports 优化,这能检测出哪些导出实际被使用了。未使用的导出如 bar
,将被安全删除。最终的产物将类似:
在 production
模式中,Rspack 也会默认分析模块是否含有副作用。如果一个模块的所有导出都未被使用且没有副作用,那么整个模块都可以被删除。我们对上面的例子做一些修改:
此时 util.js
文件中的导出都没有被使用,且可以分析出该文件没有副作用,所以 util.js
可以被整个删除。
你也可以通过 package.json
或 module.rules
来手动标记模块是否包含副作用,需要开启配置 optimization.sideEffects。
在 package.json
中,可以用 true
或 false
标记当前 package 下全部 module 是否包含副作用。
上面 package.json
表示该 package 下的所有 module 都没有副作用。
也可以用写 glob 或字符串匹配部分 module,没有被匹配到的 module 会直接视为没有副作用,因此如果手动标记 side effects,一定要确保没有标记到的 module 都没有副作用。
上述写法表示除了 ./src/main.js
和所有的 .css
文件外,其他的 module 都没有副作用。
重导出在开发中非常常见,但如果一个模块中包含重导出过多,这个模块会引入太多的其他模块,但一般我们只需要其中的一小部分,Rspack 能够优化这种场景,确保引用方可以直接访问到实际的导出模块。看以下包含重导出的例子:
Rspack 默认开启 providedExports,这可以分析出重导出模块中的所有导出以及他们各自的来源。
如果 src/re-exports.js
不包含副作用,Rspack 可以将 src/main.js
中对 src/re-exports.js
的引入,直接转换成对 src/value.js
的引入,效果类似于:
这样做的好处是可以直接忽略掉整个 src/re-exports.js
模块。
由于能够分析出 src/re-exports.js
所有的重导出,可以知道 src/value.js
中的 foo
并未使用到,最终产物中 foo
会被删掉。
有时候,即使导出被访问到,它们也可能并不会被实际使用到。例如:
在上例中,即使 log
函数和 bar
变量依赖了 foo
,由于他们自身未被使用到,foo
仍然可以被视为死代码并删除。
在开启 innerGraph 优化后(在 production
模式下默认开启),针对跨模块的复杂情形,Rspack 依然能够追踪变量的使用情况,从而实现精确的代码优化。
在这种情况下,由于 value
最终被使用,它所依赖的 foo
也得以保留。
通过 /*#__PURE__*/
注解可以告诉 Rspack 某个函数调用无副作用。它可以被放到函数调用之前,用来标记此函数调用是无副作用的。
如果一个没被使用的变量定义的初始值被认为是无副作用的,它会被标记为死代码,不会被执行且会被压缩工具清除掉。
true
时这个行为将被启用。