122嗅义、Vue Cli3 項目打包優(yōu)化

一、新建項目

使用 vue-cli3 構建一個初始的Vue項目:Cli3 官方文檔

因為使用了cli3篡石,很多目錄結構不見了芥喇,而相關配置是放在vue.config.js里面的,因此在根目錄凰萨,新建一個vue.config.js

module.exports = {}

二继控、正式優(yōu)化

1、將 productionSourceMap 設為 false

(1) 在vue.config.jsmodule.exports寫入:

module.exports = {
    productionSourceMap: false
}

2胖眷、圖片壓縮

vue正常打包之后一些圖片文件很大武通,使打包體積很大,通過image-webpack-loader插件可將大的圖片進行壓縮從而縮小打包體積

(1) 先安裝依賴:cnpm install image-webpack-loader --save-dev
(2) 在vue.config.jsmodule.exports寫入:

module.exports = {
    productionSourceMap: false,
    chainWebpack: config => {
        // ============壓縮圖片 start============
        config.module
            .rule('images')
            .use('image-webpack-loader')
            .loader('image-webpack-loader')
            .options({ bypassOnDebug: true })
            .end()
        // ============壓縮圖片 end============
    }
}

3珊搀、cdn配置(可選)

(1) 在vue.config.js 最上邊寫入:

// 是否為生產環(huán)境
const isProduction = process.env.NODE_ENV !== 'development'

// 本地環(huán)境是否需要使用cdn
const devNeedCdn = false

// cdn鏈接
const cdn = {
    // cdn:模塊名稱和模塊作用域命名(對應window里面掛載的變量名稱)
    externals: {
        vue: 'Vue',
        vuex: 'Vuex',
        'vue-router': 'VueRouter'
    },
    // cdn的css鏈接
    css: [],
    // cdn的js鏈接
    js: [
        'https://cdn.staticfile.org/vue/2.6.10/vue.min.js',
        'https://cdn.staticfile.org/vuex/3.0.1/vuex.min.js',
        'https://cdn.staticfile.org/vue-router/3.0.3/vue-router.min.js'
    ]
}

(2) 在vue.config.js module.exports chainWebpack中寫入:

// ============注入cdn start============
config.plugin('html').tap(args => {
    // 生產環(huán)境或本地需要cdn時冶忱,才注入cdn
    if (isProduction || devNeedCdn) args[0].cdn = cdn
    return args
})
// ============注入cdn start============

(3) 在vue.config.js module.exports configureWebpack中寫入:

configureWebpack: config => {
    // 用cdn方式引入,則構建時要忽略相關資源
    if (isProduction || devNeedCdn) config.externals = cdn.externals
}

(4) 當前配置的vue.config.js

// 是否為生產環(huán)境
const isProduction = process.env.NODE_ENV !== 'development'

// 本地環(huán)境是否需要使用cdn
const devNeedCdn = false

// cdn鏈接
const cdn = {
    // cdn:模塊名稱和模塊作用域命名(對應window里面掛載的變量名稱)
    externals: {
        vue: 'Vue',
        vuex: 'Vuex',
        'vue-router': 'VueRouter'
    },
    // cdn的css鏈接
    css: [],
    // cdn的js鏈接
    js: [
        'https://cdn.staticfile.org/vue/2.6.10/vue.min.js',
        'https://cdn.staticfile.org/vuex/3.0.1/vuex.min.js',
        'https://cdn.staticfile.org/vue-router/3.0.3/vue-router.min.js'
    ]
}

module.exports = {
    productionSourceMap: false,
    chainWebpack: config => {
        // ============壓縮圖片 start============
        config.module
            .rule('images')
            .use('image-webpack-loader')
            .loader('image-webpack-loader')
            .options({ bypassOnDebug: true })
            .end()
        // ============壓縮圖片 end============

        // ============注入cdn start============
        config.plugin('html').tap(args => {
            // 生產環(huán)境或本地需要cdn時境析,才注入cdn
            if (isProduction || devNeedCdn) args[0].cdn = cdn
            return args
        })
        // ============注入cdn start============
    },
    configureWebpack: config => {
        // 用cdn方式引入囚枪,則構建時要忽略相關資源
        if (isProduction || devNeedCdn) config.externals = cdn.externals
    }
}

(5) 在public index.html 寫入

<!DOCTYPE html>
<html lang="en">
    <head>
        <meta charset="utf-8" />
        <meta http-equiv="X-UA-Compatible" content="IE=edge" />
        <meta name="viewport" content="width=device-width,initial-scale=1.0" />
        <link rel="icon" href="<%= BASE_URL %>favicon.ico" />
        <!-- 使用CDN的CSS文件 -->
        <% for (var i in htmlWebpackPlugin.options.cdn &&
        htmlWebpackPlugin.options.cdn.css) { %>
        <link
            href="<%= htmlWebpackPlugin.options.cdn.css[i] %>"
            rel="stylesheet"
        />
        <% } %>
        <!-- 使用CDN的CSS文件 -->
        <title>cli3_base</title>
    </head>
    <body>
        <noscript>
            <strong
                >We're sorry but cli3_base doesn't work properly without
                JavaScript enabled. Please enable it to continue.</strong
            >
        </noscript>
        <div id="app"></div>

        <!-- 使用CDN的JS文件 -->
        <% for (var i in htmlWebpackPlugin.options.cdn &&
        htmlWebpackPlugin.options.cdn.js) { %>
        <script src="<%= htmlWebpackPlugin.options.cdn.js[i] %>"></script>
        <% } %>
        <!-- 使用CDN的JS文件 -->

        <!-- built files will be auto injected -->
    </body>
</html>

(6) 重啟項目npm run serve
(7) 在src/router.js修改

Vue.use(Router)

改為

if (!window.VueRouter) Vue.use(Router)

(8) 重新啟動npm run serve即可,現(xiàn)在的配置是開發(fā)環(huán)境劳淆,在瀏覽器的Network JS里面是看不到的链沼。若想查看,請將vue.config.js里面的

// 本地環(huán)境是否需要使用cdn
const devNeedCdn = false

改為

// 本地環(huán)境是否需要使用cdn
const devNeedCdn = true

然后再次重啟npm run serve沛鸵,然后瀏覽器查看Network JS

4括勺、代碼壓縮

(1) 安裝依賴:cnpm i -D uglifyjs-webpack-plugin
(2) 在vue.config.js 最上邊引入依賴

// 代碼壓縮
const UglifyJsPlugin = require('uglifyjs-webpack-plugin')

(3) 在vue.config.js module.exports configureWebpack 里面新增

// 生產環(huán)境相關配置
if (isProduction) {
    // 代碼壓縮
    config.plugins.push(
        new UglifyJsPlugin({
            uglifyOptions: {
                //生產環(huán)境自動刪除console
                compress: {
                    warnings: false, // 若打包錯誤,則注釋這行
                    drop_debugger: true,
                    drop_console: true,
                    pure_funcs: ['console.log']
                }
            },
            sourceMap: false,
            parallel: true
        })
    )
}

5曲掰、開啟Gzip

(1) 安裝依賴:cnpm install --save-dev compression-webpack-plugin
(2) 在vue.config.js 頂部引入依賴

// gzip壓縮
const CompressionWebpackPlugin = require('compression-webpack-plugin')

(3) 在vue.config.js module.exports configureWebpack 里面新增疾捍,直接放在代碼壓縮下邊即可

// 生產環(huán)境相關配置
if (isProduction) {
    // 代碼壓縮
    // ..................
    // gzip壓縮
    const productionGzipExtensions = ['html', 'js', 'css']
    config.plugins.push(
        new CompressionWebpackPlugin({
            filename: '[path].gz[query]',
            algorithm: 'gzip',
            test: new RegExp(
                '\\.(' + productionGzipExtensions.join('|') + ')$'
            ),
            threshold: 10240, // 只有大小大于該值的資源會被處理 10240
            minRatio: 0.8, // 只有壓縮率小于這個值的資源才會被處理
            deleteOriginalAssets: false // 刪除原文件
        })
    )
}

6、公共代碼抽離

(1) 在vue.config.js module.exports configureWebpack 里面新增栏妖,直接放在gzip壓縮下邊即可

// 公共代碼抽離
config.optimization = {
    splitChunks: {
        cacheGroups: {
            vendor: {
                chunks: 'all',
                test: /node_modules/,
                name: 'vendor',
                minChunks: 1,
                maxInitialRequests: 5,
                minSize: 0,
                priority: 100
            },
            common: {
                chunks: 'all',
                test: /[\\/]src[\\/]js[\\/]/,
                name: 'common',
                minChunks: 2,
                maxInitialRequests: 5,
                minSize: 0,
                priority: 60
            },
            styles: {
                name: 'styles',
                test: /\.(sa|sc|c)ss$/,
                chunks: 'all',
                enforce: true
            },
            runtimeChunk: {
                name: 'manifest'
            }
        }
    }
}

完整的vue.config.js

// 代碼壓縮
const UglifyJsPlugin = require('uglifyjs-webpack-plugin')

// gzip壓縮
const CompressionWebpackPlugin = require('compression-webpack-plugin')

// 是否為生產環(huán)境
const isProduction = process.env.NODE_ENV !== 'development'

// 本地環(huán)境是否需要使用cdn
const devNeedCdn = true

// cdn鏈接
const cdn = {
    // cdn:模塊名稱和模塊作用域命名(對應window里面掛載的變量名稱)
    externals: {
        vue: 'Vue',
        vuex: 'Vuex',
        'vue-router': 'VueRouter'
    },
    // cdn的css鏈接
    css: [],
    // cdn的js鏈接
    js: [
        'https://cdn.staticfile.org/vue/2.6.10/vue.min.js',
        'https://cdn.staticfile.org/vuex/3.0.1/vuex.min.js',
        'https://cdn.staticfile.org/vue-router/3.0.3/vue-router.min.js'
    ]
}

module.exports = {
    productionSourceMap: false,
    chainWebpack: config => {
        // ============壓縮圖片 start============
        config.module
            .rule('images')
            .use('image-webpack-loader')
            .loader('image-webpack-loader')
            .options({ bypassOnDebug: true })
            .end()
        // ============壓縮圖片 end============

        // ============注入cdn start============
        config.plugin('html').tap(args => {
            // 生產環(huán)境或本地需要cdn時乱豆,才注入cdn
            if (isProduction || devNeedCdn) args[0].cdn = cdn
            return args
        })
        // ============注入cdn start============
    },
    configureWebpack: config => {
        // 用cdn方式引入,則構建時要忽略相關資源
        if (isProduction || devNeedCdn) config.externals = cdn.externals

        // 生產環(huán)境相關配置
        if (isProduction) {
            // 代碼壓縮
            config.plugins.push(
                new UglifyJsPlugin({
                    uglifyOptions: {
                        //生產環(huán)境自動刪除console
                        compress: {
                            warnings: false, // 若打包錯誤吊趾,則注釋這行
                            drop_debugger: true,
                            drop_console: true,
                            pure_funcs: ['console.log']
                        }
                    },
                    sourceMap: false,
                    parallel: true
                })
            )

            // gzip壓縮
            const productionGzipExtensions = ['html', 'js', 'css']
            config.plugins.push(
                new CompressionWebpackPlugin({
                    filename: '[path].gz[query]',
                    algorithm: 'gzip',
                    test: new RegExp(
                        '\\.(' + productionGzipExtensions.join('|') + ')$'
                    ),
                    threshold: 10240, // 只有大小大于該值的資源會被處理 10240
                    minRatio: 0.8, // 只有壓縮率小于這個值的資源才會被處理
                    deleteOriginalAssets: false // 刪除原文件
                })
            )

            // 公共代碼抽離
            config.optimization = {
                splitChunks: {
                    cacheGroups: {
                        vendor: {
                            chunks: 'all',
                            test: /node_modules/,
                            name: 'vendor',
                            minChunks: 1,
                            maxInitialRequests: 5,
                            minSize: 0,
                            priority: 100
                        },
                        common: {
                            chunks: 'all',
                            test: /[\\/]src[\\/]js[\\/]/,
                            name: 'common',
                            minChunks: 2,
                            maxInitialRequests: 5,
                            minSize: 0,
                            priority: 60
                        },
                        styles: {
                            name: 'styles',
                            test: /\.(sa|sc|c)ss$/,
                            chunks: 'all',
                            enforce: true
                        },
                        runtimeChunk: {
                            name: 'manifest'
                        }
                    }
                }
            }
        }
    }
}

?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末咙鞍,一起剝皮案震驚了整個濱河市房官,隨后出現(xiàn)的幾起案子趾徽,更是在濱河造成了極大的恐慌续滋,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,123評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件孵奶,死亡現(xiàn)場離奇詭異疲酌,居然都是意外死亡,警方通過查閱死者的電腦和手機了袁,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,031評論 2 384
  • 文/潘曉璐 我一進店門朗恳,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人载绿,你說我怎么就攤上這事粥诫。” “怎么了崭庸?”我有些...
    開封第一講書人閱讀 156,723評論 0 345
  • 文/不壞的土叔 我叫張陵怀浆,是天一觀的道長。 經常有香客問我怕享,道長执赡,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,357評論 1 283
  • 正文 為了忘掉前任函筋,我火速辦了婚禮沙合,結果婚禮上,老公的妹妹穿的比我還像新娘跌帐。我一直安慰自己首懈,他們只是感情好,可當我...
    茶點故事閱讀 65,412評論 5 384
  • 文/花漫 我一把揭開白布谨敛。 她就那樣靜靜地躺著究履,像睡著了一般。 火紅的嫁衣襯著肌膚如雪佣盒。 梳的紋絲不亂的頭發(fā)上挎袜,一...
    開封第一講書人閱讀 49,760評論 1 289
  • 那天,我揣著相機與錄音肥惭,去河邊找鬼盯仪。 笑死,一個胖子當著我的面吹牛蜜葱,可吹牛的內容都是我干的全景。 我是一名探鬼主播,決...
    沈念sama閱讀 38,904評論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼牵囤,長吁一口氣:“原來是場噩夢啊……” “哼爸黄!你這毒婦竟也來了滞伟?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 37,672評論 0 266
  • 序言:老撾萬榮一對情侶失蹤炕贵,失蹤者是張志新(化名)和其女友劉穎梆奈,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體称开,經...
    沈念sama閱讀 44,118評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡亩钟,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 36,456評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了鳖轰。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片清酥。...
    茶點故事閱讀 38,599評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖蕴侣,靈堂內的尸體忽然破棺而出焰轻,到底是詐尸還是另有隱情,我是刑警寧澤昆雀,帶...
    沈念sama閱讀 34,264評論 4 328
  • 正文 年R本政府宣布辱志,位于F島的核電站,受9級特大地震影響忆肾,放射性物質發(fā)生泄漏荸频。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,857評論 3 312
  • 文/蒙蒙 一客冈、第九天 我趴在偏房一處隱蔽的房頂上張望旭从。 院中可真熱鬧,春花似錦场仲、人聲如沸和悦。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,731評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽鸽素。三九已至,卻和暖如春亦鳞,著一層夾襖步出監(jiān)牢的瞬間馍忽,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,956評論 1 264
  • 我被黑心中介騙來泰國打工燕差, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留遭笋,地道東北人。 一個月前我還...
    沈念sama閱讀 46,286評論 2 360
  • 正文 我出身青樓徒探,卻偏偏與公主長得像瓦呼,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子测暗,可洞房花燭夜當晚...
    茶點故事閱讀 43,465評論 2 348

推薦閱讀更多精彩內容

  • 第一次與傳說中的勵志人士親密接觸央串,他就是劉大銘磨澡,小小的病體不能阻擋他前進的腳步! 大銘分享了五個他生命里...
    shary江閱讀 110評論 0 0
  • 可能是因為陰雨連綿 這樣的夜是無法入睡的 隨著雨水的滴落质和,心也會無比寂寥 這種寂寥讓人不知所措 理想稳摄、鄉(xiāng)愁、生活的...
    阿莫西林_閱讀 341評論 0 0
  • 有風穿過馬路穿過孩子發(fā)熱的胸膛侦另。 這樣一個難得的季節(jié)一件薄外套就能對清晨有所交代快落地的枯葉秩命,陽光中也襯得溫柔老黃...
    魚東閱讀 294評論 1 7
  • 深秋的這樣一個黎明,無限清醒在心底袄友,遠行的我看著天空慢慢亮起來 ------《深秋的黎明》 很多時我們把精力都放在...
    凌風傳說閱讀 966評論 1 1
  • 依依是三年級的小女生殿托,長得清秀,愛跳舞剧蚣,小嘴巴吧啦吧啦愛說話支竹,是個可愛的孩子。但是挑食鸠按,挑得還挺嚴重礼搁。每次吃飯,依...
    格乃閱讀 1,344評論 15 36