vue-cli4打包最强优化(10M变300kb),网易架构师深入讲解前端开发

本文详细介绍了如何使用Vue CLI4进行项目打包优化,包括图片压缩、只打包改变的文件、公共代码抽离、配置打包分析、骨架屏实现以及完整的vue.config.js配置。通过这些技巧,可以显著减少项目打包后的大小,提高前端应用的加载速度和性能。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

压缩前后大小大致如下:

在这里插入图片描述

可以看到相应头中存在 Content-Encoding:gzip 表示已经配置成功

在这里插入图片描述

五、图片压缩

一张图片压缩前后对比:

在这里插入图片描述

需要下载 image-webpack-loader

npm install image-webpack-loader --save-dev

module.exports = {

// 根据你的实际情况更改这里

publicPath,

assetsDir: ‘assets’,

lintOnSave: true,

// image 压缩 定义在chainWebpack中

chainWebpack: config => {

config.module

.rule(‘images’)

.use(‘image-webpack-loader’)

.loader(‘image-webpack-loader’)

.options({

bypassOnDebug: true

})

.end()}

}

此插件容易下载失败,导致运行报错

  1. 若安装过 image-webpack-loader 先卸载

//npm 安装的npm 则npm 移除

npm uninstall image-webpack-loader

//如果yarn安装的,则yarn 移除

yarn remove image-webpack-loader

2、使用 cnpm , 这一步意思就是安装 cnpm 然后将全局的 registry 设置成阿里的镜像,国内阿里比较快

npm install cnpm -g --registry=https://registry.npm.taobao.org

3、使用 cnpm 安装 image-webpack-loader 会发现很快就安装好了,【手动滑稽】

cnpm install --save-dev image-webpack-loader

六、只打包改变的文件

const { HashedModuleIdsPlugin } = require(‘webpack’);

configureWebpack: config => {

const plugins = [];

plugins.push(

new HashedModuleIdsPlugin()

)

}

七、公共代码抽离

如何提取公共代码?

从webpack4开始官方移除了commonchunk插件,改用了optimization属性进行更加灵活的配置,这也应该是从V3升级到V4的代码修改过程中最为复杂的一部分

splitChunks: {

chunks: "async”,//默认作用于异步chunk,值为all/initial/async/function(chunk),值为function时第一个参数为遍历所有入口chunk时的chunk模块,chunk._modules为chunk所有依赖的模块,通过chunk的名字和所有依赖模块的resource可以自由配置,会抽取所有满足条件chunk的公有模块,以及模块的所有依赖模块,包括css

minSize: 30000, //表示在压缩前的最小模块大小,默认值是30kb

minChunks: 1, // 表示被引用次数,默认为1;

maxAsyncRequests: 5, //所有异步请求不得超过5个

maxInitialRequests: 3, //初始话并行请求不得超过3个

automaticNameDelimiter:'',//名称分隔符,默认是

name: true, //打包后的名称,默认是chunk的名字通过分隔符(默认是~)分隔

cacheGroups: { //设置缓存组用来抽取满足不同规则的chunk,下面以生成common为例

common: {

name: ‘common’, //抽取的chunk的名字

chunks(chunk) { //同外层的参数配置,覆盖外层的chunks,以chunk为维度进行抽取

},

test(module, chunks) { //可以为字符串,正则表达式,函数,以module为维度进行抽取,只要是满足条件的module都会被抽取到该common的chunk中,为函数时第一个参数是遍历到的每一个模块,第二个参数是每一个引用到该模块的chunks数组。自己尝试过程中发现不能提取出css,待进一步验证。

},

priority: 10, //优先级,一个chunk很可能满足多个缓存组,会被抽取到优先级高的缓存组中

minChunks: 2, //最少被几个chunk引用

reuseExistingChunk: true,// 如果该chunk中引用了已经被抽取的chunk,直接引用该chunk,不会重复打包代码

enforce: true // 如果cacheGroup中没有设置minSize,则据此判断是否使用上层的minSize,true:则使用0,false:使用上层minSize

}

}

}

公共模块抽离

举例:

项目中分别有a.js, b.js, page1.js, page2.js这四个JS文件, page1.js 和

page2.js中同时都引用了a.js, b.js, 这时候想把a.js, b.js抽离出来合并成一个公共的js,然后在page1,page2中自动引入这个公共的js,怎么配置呢?

第三方模块抽离

页面中有时会引入第三方模块,比如import $ from ‘jquery’;

page1中需要引用,page2中也需要引用,这时候就可以用vendor把jquery抽离出来,

如下:

// 公共代码抽离

configureWebpack: config => {

//…

//优化项配置

config.optimization = {

splitChunks: { // 分割代码块

cacheGroups: {

vendor: {//第三方库抽离

chunks: ‘all’,

test: /node_modules/,

name: ‘vendor’,

minChunks: 1,//在分割之前,这个代码块最小应该被引用的次数

maxInitialRequests: 5,

minSize: 0,//大于0个字节

priority: 100//权重

},

common: { //公用模块抽离

chunks: ‘all’,

test: /[\/]src[\/]js[\/]/,

name: ‘common’,

minChunks: 2,在分割之前,这个代码块最小应该被引用的次数

maxInitialRequests: 5,

minSize: 0,//大于0个字节

priority: 60

},

styles: { //样式抽离

name: ‘styles’,

test: /.(sa|sc|c)ss$/,

chunks: ‘all’,

enforce: true

},

runtimeChunk: {

name: ‘manifest’

}

}

}

}

}

八、配置 打包分析

安装`cnpm i webpack-bundle-analyzer -D

const BundleAnalyzerPlugin = require(‘webpack-bundle-analyzer’).BundleAnalyzerPlugin

module.exports = {

chainWebpack: config => {

// 打包分析

if (IS_PROD) {

config.plugin(‘webpack-report’).use(BundleAnalyzerPlugin, [

{

analyzerMode: ‘static’

}

])

}

}

}

九、骨架屏

安装插件npm install vue-skeleton-webpack-plugin

在src下新建Skeleton文件夹,其中新建index.js以及index.vue,在其中写入以下内容,其中,骨架屏的index.vue页面样式请自行编辑

index.js

import Vue from ‘vue’

import home from ‘./index.vue’

import list from ‘./list.vue’

export default new Vue({

components: {

home,

list

},

template: `

`

})

index.vue(骨架屏页面) list.vue同理

vue.config.js 配置

//骨架屏渲染

const SkeletonWebpackPlugin = require(‘vue-skeleton-webpack-plugin’)

//path引入

const path = require(‘path’)

//configureWebpack模块中写入内容

// 骨架屏渲染

config.plugins.push(new SkeletonWebpackPlugin({

webpackConfig: {

entry: {

app: path.join(__dirname, ‘./src/Skeleton/index.js’),

},

},

minimize: true,

quiet: true,

// 如果不设置那么所有的路由都会共享这个骨架屏组件

router: {

mode: ‘hash’,

// 给对应的路由设置对应的骨架屏组件,skeletonId的值根据组件设置的id

routes: [

{ path: ‘/home’, skeletonId: ‘home’ },

{ path: ‘/list’, skeletonId: ‘list’ },

]

}))

七、完整配置

vue.config.js完整配置

const path = require(‘path’);

const UglifyJsPlugin = require(‘uglifyjs-webpack-plugin’) // 去掉注释

const CompressionWebpackPlugin = require(‘compression-webpack-plugin’); // 开启压缩

const { HashedModuleIdsPlugin } = require(‘webpack’);

function resolve(dir) {

return path.join(__dirname, dir)

}

const isProduction = process.env.NODE_ENV === ‘production’;

// cdn预加载使用

const externals = {

‘vue’: ‘Vue’,

‘vue-router’: ‘VueRouter’,

‘vuex’: ‘Vuex’,

‘axios’: ‘axios’,

“element-ui”: “ELEMENT”

}

const cdn = {

// 开发环境

dev: {

css: [

‘https://unpkg.com/element-ui/lib/theme-chalk/index.css’

],

js: []

},

// 生产环境

build: {

css: [

‘https://unpkg.com/element-ui/lib/theme-chalk/index.css’

],

js: [

‘https://cdn.jsdelivr.net/npm/vue@2.5.17/dist/vue.min.js’,

‘https://cdn.jsdelivr.net/npm/vue-router@3.0.1/dist/vue-router.min.js’,

‘https://cdn.jsdelivr.net/npm/vuex@3.0.1/dist/vuex.min.js’,

‘https://cdn.jsdelivr.net/npm/axios@0.18.0/dist/axios.min.js’,

‘https://unpkg.com/element-ui/lib/index.js’

]

}

}

module.exports = {

lintOnSave: false, // 关闭eslint

productionSourceMap: false,

publicPath: ‘./’,

outputDir: process.env.outputDir, // 生成文件的目录名称

chainWebpack: config => {

config.resolve.alias

.set(‘@’, resolve(‘src’))

// 压缩图片

config.module

.rule(‘images’)

.test(/.(png|jpe?g|gif|svg)(?.*)?$/)

.use(‘image-webpack-loader’)

.loader(‘image-webpack-loader’)

.options({ bypassOnDebug: true })

// webpack 会默认给commonChunk打进chunk-vendors,所以需要对webpack的配置进行delete

config.optimization.delete(‘splitChunks’)

config.plugin(‘html’).tap(args => {

if (process.env.NODE_ENV === ‘production’) {

args[0].cdn = cdn.build

}

if (process.env.NODE_ENV === ‘development’) {

args[0].cdn = cdn.dev

}

return args

})

config

.plugin(‘webpack-bundle-analyzer’)

.use(require(‘webpack-bundle-analyzer’).BundleAnalyzerPlugin)

},

configureWebpack: config => {

const plugins = [];

if (isProduction) {

plugins.push(

new UglifyJsPlugin({

uglifyOptions: {

output: {

comments: false, // 去掉注释

},

warnings: false,

compress: {

drop_console: true,

drop_debugger: false,

pure_funcs: [‘console.log’]//移除console

}

}

})

)

// 服务器也要相应开启gzip

plugins.push(

new CompressionWebpackPlugin({

algorithm: ‘gzip’,

test: /.(js|css)$/,// 匹配文件名

threshold: 10000, // 对超过10k的数据压缩

deleteOriginalAssets: false, // 不删除源文件

minRatio: 0.8 // 压缩比

})

)

// 用于根据模块的相对路径生成 hash 作为模块 id, 一般用于生产环境

plugins.push(

new HashedModuleIdsPlugin()

)

// 开启分离js

config.optimization = {

runtimeChunk: ‘single’,

splitChunks: {

chunks: ‘all’,

maxInitialRequests: Infinity,

minSize: 1000 * 60,

cacheGroups: {

vendor: {

test: /[\/]node_modules[\/]/,

name(module) {

// 排除node_modules 然后吧 @ 替换为空 ,考虑到服务器的兼容

const packageName = module.context.match(/[\/]node_modules\/([\/]|$)/)[1]

return npm.${packageName.replace('@', '')}

}

}

}

}

};

// 取消webpack警告的性能提示

config.performance = {

hints: ‘warning’,

//入口起点的最大体积

maxEntrypointSize: 1000 * 500,

//生成文件的最大体积

maxAssetSize: 1000 * 1000,

//只给出 js 文件的性能提示

assetFilter: function (assetFilename) {

return assetFilename.endsWith(‘.js’);

}

}

// 打包时npm包转CDN

config.externals = externals;

}

return { plugins }

},

pluginOptions: {

// 配置全局less

‘style-resources-loader’: {

preProcessor: ‘less’,

patterns: [resolve(‘./src/style/theme.less’)]

}

},

devServer: {

open: false, // 自动启动浏览器

host: ‘0.0.0.0’, // localhost

port: 6060, // 端口号

https: false,

hotOnly: false, // 热更新

proxy: {

‘^/sso’: {

target: process.env.VUE_APP_SSO, // 重写路径

ws: true, //开启WebSocket

secure: false, // 如果是https接口,需要配置这个参数

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数前端工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Web前端开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上前端开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip1024c (备注前端)
img

学习笔记

主要内容包括html,css,html5,css3,JavaScript,正则表达式,函数,BOM,DOM,jQuery,AJAX,vue等等

HTML/CSS

CodeChina开源项目:【大厂前端面试题解析+核心总结学习笔记+真实项目实战+最新讲解视频】

**HTML:**HTML基本结构,标签属性,事件属性,文本标签,多媒体标签,列表 / 表格 / 表单标签,其他语义化标签,网页结构,模块划分

**CSS:**CSS代码语法,CSS 放置位置,CSS的继承,选择器的种类/优先级,背景样式,字体样式,文本属性,基本样式,样式重置,盒模型样式,浮动float,定位position,浏览器默认样式

HTML5 /CSS3

**HTML5:**HTML5 的优势,HTML5 废弃元素,HTML5 新增元素,HTML5 表单相关元素和属性

**CSS3:**CSS3 新增选择器,CSS3 新增属性,新增变形动画属性,3D变形属性,CSS3 的过渡属性,CSS3 的动画属性,CSS3 新增多列属性,CSS3新增单位,弹性盒模型

JavaScript

**JavaScript:**JavaScript基础,JavaScript数据类型,算术运算,强制转换,赋值运算,关系运算,逻辑运算,三元运算,分支循环,switch,while,do-while,for,break,continue,数组,数组方法,二维数组,字符串

一个人可以走的很快,但一群人才能走的更远。如果你从事以下工作或对以下感兴趣,欢迎戳这里加入程序员的圈子,让我们一起学习成长!

AI人工智能、Android移动开发、AIGC大模型、C C#、Go语言、Java、Linux运维、云计算、MySQL、PMP、网络安全、Python爬虫、UE5、UI设计、Unity3D、Web前端开发、产品经理、车载开发、大数据、鸿蒙、计算机网络、嵌入式物联网、软件测试、数据结构与算法、音视频开发、Flutter、IOS开发、PHP开发、.NET、安卓逆向、云计算

)**

**HTML:**HTML基本结构,标签属性,事件属性,文本标签,多媒体标签,列表 / 表格 / 表单标签,其他语义化标签,网页结构,模块划分

**CSS:**CSS代码语法,CSS 放置位置,CSS的继承,选择器的种类/优先级,背景样式,字体样式,文本属性,基本样式,样式重置,盒模型样式,浮动float,定位position,浏览器默认样式

[外链图片转存中…(img-BF91qFgk-1712195063013)]

HTML5 /CSS3

**HTML5:**HTML5 的优势,HTML5 废弃元素,HTML5 新增元素,HTML5 表单相关元素和属性

**CSS3:**CSS3 新增选择器,CSS3 新增属性,新增变形动画属性,3D变形属性,CSS3 的过渡属性,CSS3 的动画属性,CSS3 新增多列属性,CSS3新增单位,弹性盒模型

[外链图片转存中…(img-U2rTysCu-1712195063014)]

JavaScript

**JavaScript:**JavaScript基础,JavaScript数据类型,算术运算,强制转换,赋值运算,关系运算,逻辑运算,三元运算,分支循环,switch,while,do-while,for,break,continue,数组,数组方法,二维数组,字符串

[外链图片转存中…(img-oXSMbqeI-1712195063014)]

一个人可以走的很快,但一群人才能走的更远。如果你从事以下工作或对以下感兴趣,欢迎戳这里加入程序员的圈子,让我们一起学习成长!

AI人工智能、Android移动开发、AIGC大模型、C C#、Go语言、Java、Linux运维、云计算、MySQL、PMP、网络安全、Python爬虫、UE5、UI设计、Unity3D、Web前端开发、产品经理、车载开发、大数据、鸿蒙、计算机网络、嵌入式物联网、软件测试、数据结构与算法、音视频开发、Flutter、IOS开发、PHP开发、.NET、安卓逆向、云计算

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值