webpack學(xué)習(xí)第十二步——代碼分割

打包命令

  • 為了development模式下也能很好觀察打包的文件彻磁,我們?cè)趐ackage.json中增加一個(gè)打包命令舶得,不使用devServer
"scripts": {
    "dev": "webpack --config ./config/webpack.dev.js",
    "start": "webpack-dev-server --config ./config/webpack.dev.js",
    "build": "webpack --config ./config/webpack.prod.js"
}

index.js引入lodash

  • 安裝lodash
npm install lodash -D
  • index.js使用lodash
import _ from  'lodash'

console.log(_.join(['a','b','c'],'***'))
  • 打包后生成的文件很大
  • 打包文件會(huì)很大显押,加載時(shí)間會(huì)很長(zhǎng),改了業(yè)務(wù)代碼投蝉,重新訪問(wèn)我們的頁(yè)面膊爪,又要重新加載這么大的文件內(nèi)容

寫代碼時(shí)的代碼分割

  • 新建一個(gè)lodash.js文件
// 導(dǎo)入import 文件,并且將lodash掛載到window對(duì)象上
import _ from 'lodash'
window._ = _
  • 修改index.js
// 不需要import lodash管毙,可直接使用_
console.log(_.join(['a','b','c'],'***'))
  • 修改webpack.common.js
// 打包的入口文件改為兩個(gè)腿椎,單獨(dú)打包lodash
entry: {
    lodash: './src/lodash.js',
    main: './src/index.js',
},
  • main.js被拆成loadash.js 和 mian.js(業(yè)務(wù)代碼),瀏覽器并行加載兩個(gè)文件夭咬,當(dāng)業(yè)務(wù)代碼發(fā)生變化啃炸,用戶不需加載lodash,只需加載main.js

利用webpack實(shí)現(xiàn)代碼分割

  • 不需要開發(fā)者自己再寫個(gè)lodash.js將lodash掛載到window卓舵,而是自動(dòng)把類庫(kù)拆分成一個(gè)文件
  • 修改index.js
// 導(dǎo)入lodash
import _ from 'lodash'
console.log(_.join(['a','b','c'],'***'))
  • 修改webpack.common.js
const path = require('path')
const HtmlWebpackPlugin = require('html-webpack-plugin')
const CleanWebpackPlugin = require('clean-webpack-plugin')

module.exports = {
    entry: {
         // 打包index.js文件
        main: './src/index.js',
    },
    module: {
        rules: [{
            test: /\.(jpg|png|gif)$/,
            use: {
                loader: "url-loader",
                options: {
                    name: '[name]_[hash].[ext]',
                    outputPath: 'images/',
                    limit: 2048
                }
            }
        },{
            test: /\.css$/,
            use: [
                'style-loader',
                'css-loader',
                'postcss-loader'
            ]
        },{
            test: /\.js$/,
            exclude: /node_modules/,
            loader: 'babel-loader',
        }]
    },
    plugins: [
        new HtmlWebpackPlugin({
            template: 'src/index.html'
        }),
        new CleanWebpackPlugin({
            root: path.resolve(__dirname, '../')
        }),
    ],
    output: {
        path: path.resolve(__dirname, '../build'),
        filename: '[name].js'
    },
    optimization: {
    // 設(shè)置splitChunks
        splitChunks: {
            chunks: "all"
        }
    }
}
  • 打包后index.js會(huì)自動(dòng)變?yōu)閮蓚€(gè)文件南用,實(shí)現(xiàn)了自動(dòng)的代碼分割

異步import的代碼分割

  • 異步import需要引入官方的babel
npm install @babel/plugin-syntax-dynamic-import -D
  • 修改babelrc
{
  "presets": [
    [
      "@babel/preset-env",{
        "targets": {
          "chrome": "67"
        },
        "useBuiltIns": "usage"
      }
    ],
    "@babel/preset-react"
  ],
  // 使用異步 import 插件
  "plugins": ["@babel/plugin-syntax-dynamic-import"]
}

  • 修改index.js為異步import方式
function getComponent() {
    return import('lodash').then(({ default: _ }) => {
        var element = document.createElement('div')
        element.innerHTML =_.join(['a','b','c'],'***')
        return element
    })
}
getComponent().then(element => document.body.appendChild(element))
  • 打包index.js會(huì)拆分成兩個(gè)文件
  • 打開index.html
  • 可以指定生成的chunk文件的文件名
function getComponent() {
// 通過(guò)這種注解生成chunkname,將生成的chunk文件指定文件名為lodash
    return import(/* webpackChunkName:'lodash' */'lodash').then(( _ ) => {
        var element = document.createElement('div')
        element.innerHTML =_.join(['a','b','c'],'***')
        return element
    })
}
getComponent().then(element => document.body.appendChild(element))

splitChunks配置項(xiàng)

  • 修改webpack.common.js
optimization: {
    splitChunks: {
        chunks: "all",
        cacheGroups: {
            vendors: false,
            default: false
        }
    }
}
  • 打包生成的文件不再用vendors前綴

splitChunks默認(rèn)配置項(xiàng)

optimization: {
    // splitChunks: {},
    // 等價(jià)于
    splitChunks: {
        chunks: 'async',
        minSize: 30000,
        maxSize: 0,
        minChunks: 1,
        maxAsyncRequests: 5,
        maxInitialRequests: 3,
        automaticNameDelimiter: '~',
        name: true,
        cacheGroups: {
            vendors: {
                test: /[\\/]node_modules[\\/]/,
                priority: -10
            },
            default: {
                minChunks: 2,
                priority: -20,
                reuseExistingChunk: true
            }
        }
    }
}

chunks配置

  • 默認(rèn)為async

    // 代碼分割時(shí)只對(duì)異步代碼生效掏湾,同步 import lodash 不進(jìn)行代碼分割
    chunks: 'async'
    
    • 避免受其他因素干擾我們將cacheGroups都先置為false,minSize 和 maxSize 都設(shè)置為0
    splitChunks: {
        chunks: 'async',
        minSize: 0,
        maxSize: 0,
        minChunks: 1,
        maxAsyncRequests: 5,
        maxInitialRequests: 3,
        automaticNameDelimiter: '~',
        name: true,
        cacheGroups: {
            // vendors: {
            //     test: /[\\/]node_modules[\\/]/,
            //     priority: -10,
            //     filename: "vendors.js"
            // },
            // default: {
            //     // minChunks: 2,
            //     priority: -20,
            //     reuseExistingChunk: true,
            //     filename: "common.js"
            // }
            vendors: false,
            default: false
        }
    }
    
    • 修改index.js為同步import方式
    import _ from 'lodash'
    
    var element = document.createElement('div')
    element.innerHTML =_.join(['a','b','c'],'***')
    document.body.appendChild(element)
    
    • 打包不會(huì)進(jìn)行代碼分割
    • 修改index.js為異步import方式
    function getComponent() {
        return import(/* webpackChunkName:'lodash' */'lodash').then(( _ ) => {
            var element = document.createElement('div')
            element.innerHTML =_.join(['a','b','c'],'***')
            return element
        })
    }
    getComponent().then(element => document.body.appendChild(element))
    
    • 打包時(shí)會(huì)進(jìn)行代碼分割
  • 對(duì)同步異步都進(jìn)行打包 (initial為對(duì)同步代碼做分割)

    chunks: 'all'
    
    • 需要配合cacheGroups選項(xiàng),否則同步import 仍舊不會(huì)進(jìn)行代碼分割
    cacheGroups: {
        vendors: {
        // node_modules里面的文件分割打到單獨(dú)文件里
            test: /[\\/]node_modules[\\/]/,
            priority: -10
        },
        default: false
    }
    
    • 打包后代碼被分割裹虫,vendors~main表示打包了vendors文件,入口是main
    • 可以修改vendors文件的文件名
    vendors: {
        test: /[\\/]node_modules[\\/]/,
        priority: -10,
        // 文件名為vendors.js(異步import不可用)
        filename: "vendors.js"
    },
    
    • 打包后文件名為vendors.js
  • 非node_module庫(kù)的import

    • 新建a.js
    export default {
        num : 1
    }
    
    • index.js導(dǎo)入a.js
    import a from './a'
    console.log(a.num)
    
    • 符合代碼分割后會(huì)走到cacheGroups融击,因?yàn)椴辉趎ode_modules里筑公,所以不會(huì)走vendros,而是走到default配置砚嘴,因此我們需要修改cacheGroups
    cacheGroups: {
        vendors: {
            test: /[\\/]node_modules[\\/]/,
            priority: -10,
            filename: "vendors.js"
        },
        default: {
            priority: -20,
            reuseExistingChunk: true,
        }
    }
    
    • 打包后a.js會(huì)被打包到default~main.js中
    • default設(shè)置filename
    default: {
        priority: -20,
        reuseExistingChunk: true,
        filename: "common.js"
    }
    

minSize 配置

  • 默認(rèn)引入的模塊大于30000個(gè)字節(jié)才做split
minSize: 30000
  • 上例中引入a.js十酣,設(shè)置minSize后打包,就不會(huì)進(jìn)行代碼分割
  • 將minSize改為minSize: 1后打包會(huì)進(jìn)行代碼分割际长,生成common.js文件

maxSize 配置

  • 配置maxSize,限定最大值,超過(guò)就會(huì)進(jìn)行二次拆分
// 引入的模塊大于1個(gè)字節(jié)才做split
minSize: 1,
// 引入的a.js限定了最大值兴泥,超過(guò)就會(huì)對(duì)文件進(jìn)行二次拆分
maxSize: 2,
  • 打包后commin.js被二次拆分
  • 一般不使用

minChunks

  • minChunks表示一個(gè)代碼至少被引用過(guò)多少次才進(jìn)行代碼分割
  • 修改minChunks的值
minChunks: 2
  • index.js引用lodash

  • 打包的代碼不會(huì)進(jìn)行代碼分割

  • 新建other.js工育,在改文件中也引用lodash
  • webpack.common.js中對(duì)other.js也進(jìn)行打包操作
entry: {
    main: './src/index.js',
    other: './src/other.js'
}
  • 此時(shí)lodash會(huì)被分割出來(lái),vendorsmainother表示main和other都引用了該文件

多個(gè)文件

  • main.js引入了lodash,other.js引入了lodash和jquery, a.js引入了jquery,這個(gè)時(shí)候如果cacheGroups還設(shè)定fileName會(huì)發(fā)生沖突
  • 將filename改為name,lodash和jquery都打到了vendors.js文件中
vendors: {
    test: /[\\/]node_modules[\\/]/,
    priority: -10,
    name: "vendors"
}
  • 注釋掉name,會(huì)分割為vendors~a~other.jsvendors~main~other.js
vendors: {
    test: /[\\/]node_modules[\\/]/,
    priority: -10,
    //name: "vendors"
}

其他配置

  • 其他配置
// 同時(shí)加載的模塊數(shù)最多5個(gè)搓彻,超過(guò)5個(gè)不做代碼分割
maxAsyncRequests: 5,
// 入口文件引入的文件最多三個(gè)代碼分割
maxInitialRequests: 3,
// 生成文件名稱連接符
automaticNameDelimiter: '~',
// cacheGroups 里 filename 有效
name: true,
  • cacheGroups的配置
default: {
    // minChunks: 2,
    // 優(yōu)先級(jí)低于-10如绸,所以node_modules里面用vendors
    priority: -20,
    // 一個(gè)模塊已經(jīng)被引用過(guò)嘱朽,就直接復(fù)用,不用再打包
    reuseExistingChunk: true,
    filename: "common.js"
},
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末怔接,一起剝皮案震驚了整個(gè)濱河市搪泳,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌扼脐,老刑警劉巖岸军,帶你破解...
    沈念sama閱讀 219,490評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異瓦侮,居然都是意外死亡艰赞,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,581評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門肚吏,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)方妖,“玉大人,你說(shuō)我怎么就攤上這事罚攀〉趁伲” “怎么了?”我有些...
    開封第一講書人閱讀 165,830評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵斋泄,是天一觀的道長(zhǎng)杯瞻。 經(jīng)常有香客問(wèn)我,道長(zhǎng)是己,這世上最難降的妖魔是什么又兵? 我笑而不...
    開封第一講書人閱讀 58,957評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮卒废,結(jié)果婚禮上沛厨,老公的妹妹穿的比我還像新娘。我一直安慰自己摔认,他們只是感情好逆皮,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,974評(píng)論 6 393
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著参袱,像睡著了一般电谣。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上抹蚀,一...
    開封第一講書人閱讀 51,754評(píng)論 1 307
  • 那天剿牺,我揣著相機(jī)與錄音,去河邊找鬼环壤。 笑死晒来,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的郑现。 我是一名探鬼主播湃崩,決...
    沈念sama閱讀 40,464評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼荧降,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了攒读?” 一聲冷哼從身側(cè)響起朵诫,我...
    開封第一講書人閱讀 39,357評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎薄扁,沒(méi)想到半個(gè)月后剪返,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,847評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡泌辫,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,995評(píng)論 3 338
  • 正文 我和宋清朗相戀三年随夸,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片震放。...
    茶點(diǎn)故事閱讀 40,137評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡宾毒,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出殿遂,到底是詐尸還是另有隱情诈铛,我是刑警寧澤,帶...
    沈念sama閱讀 35,819評(píng)論 5 346
  • 正文 年R本政府宣布墨礁,位于F島的核電站幢竹,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏恩静。R本人自食惡果不足惜焕毫,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,482評(píng)論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望驶乾。 院中可真熱鬧邑飒,春花似錦、人聲如沸级乐。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,023評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)风科。三九已至撒轮,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間贼穆,已是汗流浹背题山。 一陣腳步聲響...
    開封第一講書人閱讀 33,149評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留故痊,地道東北人臀蛛。 一個(gè)月前我還...
    沈念sama閱讀 48,409評(píng)論 3 373
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像崖蜜,于是被迫代替她去往敵國(guó)和親浊仆。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,086評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容

  • 前言 開發(fā)多頁(yè)應(yīng)用的時(shí)候豫领,如果不對(duì)webpack打包進(jìn)行優(yōu)化抡柿,當(dāng)某個(gè)模塊被多個(gè)入口模塊引用時(shí),它就會(huì)被打包多次(在...
    champyin閱讀 467評(píng)論 1 0
  • 如果要徹底明白 Webpack如何抽取公共代碼等恐,就要設(shè)計(jì)一個(gè)場(chǎng)景來(lái)支持抽取公共代碼的多種形式洲劣,能夠從代碼運(yùn)行的結(jié)果...
    VivaLaVida_692c閱讀 1,721評(píng)論 0 0
  • 好久沒(méi)有寫日記了。课蔬。囱稽。今天我專心學(xué)習(xí)webpack,跟著網(wǎng)上的文章請(qǐng)練完這16個(gè)webpack小例子翻譯阮一峰大神...
    還有誰(shuí)叫李狗蛋閱讀 5,108評(píng)論 1 6
  • 全局安裝webpack 全局安裝webpack會(huì)有個(gè)問(wèn)題二跋,就是當(dāng)你有兩個(gè)項(xiàng)目依賴于不同版本的webpack战惊,就會(huì)有...
    説好的妹紙呢閱讀 1,821評(píng)論 0 11
  • 這里有一個(gè)咖啡館,名字叫做“為什么咖啡館”扎即。 這里沒(méi)有關(guān)于人生的萬(wàn)能解答吞获,但有金錢絕對(duì)買不到的東西。 這里不賣答案...
    不讀書書店閱讀 1,923評(píng)論 2 4