大廠喜歡問的TreeShaking到底是啥?好在哪呢?五分鐘弄懂!
背景
大家平時在查 webpack構(gòu)建體積優(yōu)化 ,可能都會查到 tree-shaking 這個東西,很多人看到這個東西,就會把它背下來,用來應(yīng)付以后面試官可能會問到的情況。
但是,又有多少人去真的了解一下 tree-shaking 呢?自己去實踐一下看 tree-shaking 到底起了哪些作用?對于我們的打包體積的優(yōu)化又有多少呢?
有啥用?
Tree Shaking中文含義是搖樹,在webpack中指的是打包時把無用的代碼搖掉,以優(yōu)化打包結(jié)果。
而webpack5已經(jīng)自帶了這個功能了,當打包環(huán)境為production時,默認開啟tree-shaking功能。
實踐
前置準備
準備兩個文件main.js、util.js
- util.js
function a () {
console.log('a')
}
function b () {
console.log('b')
}
export default {
a, b
}
- main.js
import a from './util'
// 使用a變量,調(diào)用文件里面的a函數(shù),不使用b函數(shù)
console.log(a.a())
console.log('hello world')
// 不可能執(zhí)行的代碼
if (false) {
console.log('haha')
}
// 定義了但是沒用的變量
const m = 1
打包
前面說了webpack5在環(huán)境production下打包的話,默認開啟tree-shaking,那我們運行npm run build進行一下打包,看看打包后的代碼長啥樣:
(()=>{"use strict";
const o=function(){console.log("a")};
console.log(o())
console.log("hello world")}
)();
結(jié)論:可以看到打包后,把b函數(shù)、不可能執(zhí)行的代碼、定義未用的變量通通都剔除了,這在一個項目中,能減少很多的代碼量,進而減少打包后的文件體積。
sideEffects
副作用
先來講講一個東西——副作用,是什么東西呢?副作用指的是:除了導(dǎo)出成員之外所做的事情,我舉個例子,下面的a.js是沒副作用的,b.js是有副作用的:
- a.js
function console () {
console.log('console')
}
export default {
console
}
- b.js
function console () {
console.log('console')
}
// 這個就是副作用,會影響全局的數(shù)組
Array.prototype.func = () => {}
export default {
console
}
有無副作用的判斷,可以決定tree-shaking的優(yōu)化程度,舉個例子:
- 我現(xiàn)在引入a.js但是我不用他的console函數(shù),那么在優(yōu)化階段我完全可以不打包a.js這個文件。
- 我現(xiàn)在引入b.js但是我不用他的console函數(shù),但是我不可以不打包b.js這個文件,因為他有副作用,不能不打包。
sideEffects的使用
sideEffects可以在package.json中設(shè)置:
// 所有文件都有副作用,全都不可 tree-shaking
{
"sideEffects": true
}
// 沒有文件有副作用,全都可以 tree-shaking
{
"sideEffects": false
}
// 只有這些文件有副作用,
// 所有其他文件都可以 tree-shaking,
// 但會保留這些文件
{
"sideEffects": [
"./src/file1.js",
"./src/file2.js"
]
}
優(yōu)化體積
當我把sideEffects設(shè)置成true之后,整個打包體積增加了100k,說明默認的false還是有用的。