哈喽大家好,我是Fine。
性能优化的文章我们也分享过不少,很多同学对性能优化的理解仅限面试时背的八股文,没有在实际工作中应用。今天分享的这篇文章就是作者结合实际工作,通过发现问题,分析问题,再针对性的去解决问题。如果恰好你的项目也存在这样的问题,不妨也来试试吧。
以下是正文:
需求背景
从第三方采购的vue2 + ElementUI实现的云管平台,乙方说2011年左右就开始有这个项目了(那时候有Vue了吗,思考.jpg)。十几年的项目,我何德何能可以担此责任。里面的代码经过多人多年迭代可以用惨不忍睹来形容,吐槽归吐槽,混口饭吃,多烂的代码都得啃下去。
有一天领导找到我,问我怎么回事,打开页面需要十几秒时间也太慢了,后台管理系统不要求首屏加载时间都没有这么慢,这个对外的系统超过1秒打开时间,都会流失很多客户,不优化好年终自己看着办吧。
什么?影响年终?好的领导,我马上抽时间解决(🐂🐴)。
如何使用Lighthouse?
F12打开控制台 - Lighthouse
如下图所示选择,然后点击Anlyze page load就可以了
这里只关注性能所以只勾选了Performance指标,其他可访问性、SEO有需求一同检测的自行勾选上。
如何使用webpack-bundle-analyzer?
npm install --save-dev webpack-bundle-analyzer安装依赖
js
复制代码
// webpack.config.js配置文件
const BundleAnalyzerPlugin = require('webpack-bundle-analyzer').BundleAnalyzerPlugin;
module.exports = {
// ... 其他配置 ...
plugins: [
new BundleAnalyzerPlugin()
]
};
这样在打包的时候会生成一个静态网站查看各个模块占用的存储空间大小
复制代码
LightHouse性能指标解释:
FCP:衡量的是打开网页后,浏览器渲染第一段 DOM 内容所用的时间
LCP:用于测量视口中最大的内容元素何时渲染到屏幕上。这粗略地估算出网页主要内容何时对用户可见。
因为要优化的页面没有像官网那样有轮播图占据大量显示位置的元素,所以我主要关注FCP,FCP解决LCP也会相应变快
makefile
复制代码
Stat: 源代码阶段
Parsed: 经过webpack打包后的大小
Gzipped: 经过Gzip压缩过后的大小,实际浏览器接收的大小,需要服务器开启Gzip压缩
这里的大小主要关注chunk-vendors.js和app.js。其他都跟首页加载关系不大。下面的首屏代码大小是这两个js代码大小之和。
chunk-vendors是引用的第三方库如element-ui、echarts、vue等等打包后的代码。app.js是项目的代码。
总大小 | 首屏代码大小 | |
---|---|---|
Stat | 76.5M | 21.1M |
Parsed | 80.5M | 24.2M |
Gzipped | 13.5M | 5.9M |
这里只放Gzipped的大小,全部都放图太多了。
总大小 | 首屏代码大小 | |
---|---|---|
Stat | 30.2M | 15.2M |
Parsed | 16.6M | 6.2M |
Gzipped | 4.3M | 1.9M |
看到这有人说,虽然是快了13.6秒,但是还是要2.2s,还是不能秒开,你的年终还是不保啊。
上面Lighthouse是不会用到缓存去检测性能的,为了有效他每次检测都相当于首次全部加载,打包后的css、js静态文件都是可缓存的。用户第二次打开时,是可以做到秒开。(年终有啦-.-)
这一点在优化官网的时候很有用!大多数官网代码都十分精简,首屏加载慢大多都是因为轮播图没压缩,文件太大请求慢导致的,用压缩工具压缩一下,或者让UI换个不失真占用空间小的即可解决问题。
在用webpack-bundle-analyzer查看包大小的时候,发现一个SVG竟然有1.5M!你敢信?而且SVG不会经过打包有大小变化,就是即使经过Webpack打包,Gzip压缩,他也会占用1.5M的的大小,优化前也才5.9M,所以1.5M占比很大需要优化。
为什么会有1.5M的SVG?
图片可以转SVG,但是只是粗暴的将图片的base64编码塞到SVG里面,体积增大33%
图片未经压缩,原图比较大(甩锅设计,哈哈哈)
最后改为使用静态图片引用,而且不是首页需要用到的SVG,所以首页代码大大减少
从下图可以看到支付宝相关的SVG都比较大,最后都被我用阿里的iconfont用更小的svg替换了,减少了2.5M左右的大小!
删掉无用路由很有用!加上树摇,可以去掉很多代码。正常来说公司自研的项目,每一个路由都是必要。但是这是第三方经过多人多年迭代的项目,很多路由都是没用的。
还有一些库,在main.js文件里注册了Vue的全局组件,但是搜索整个项目根本没有用到,而且这个库还挺大的。又可以减少一些代码。
这里有个奇葩的点是,webpack@4.46.0的版本必须要指定mode: 'production'才会启用树摇!否则打包大小基本和源代码大小一样,参考上面优化前源代码76.5M,打包后80.5M代码还多了一点!(也有可能树摇开启成功了,只是启用了production其他优化减少了体积,有没有大佬指导一下!)
import Home from '@views/Home.vue'
const router = [
// 首页不要异步,才用导入的方式打包到app.js,优先加载
{
path: '/home',
component: Home
},
// 其他组件异步加载,多个小组件可以打包到一起,减少请求数,代码分离要恰到好处
{
path: '/xx',
component: () => import(/* webpackChunkName: "xx" */ '@/views/xx.vue')
}
],
项目的index.html总有一些奇奇怪怪的js和css引入如下所示,会阻塞页面的解析,我们在前端首页解析完后(DOMContentLoaded事件)加载它们。
<link
rel="stylesheet"
href="./luckysheet/plugins/css/pluginsCss.css" />
<link
rel="stylesheet"
href="./luckysheet/plugins/plugins.css" />
<link
rel="stylesheet"
href="./luckysheet/css/luckysheet.css" />
<link
rel="stylesheet"
href="./luckysheet/assets/iconfont/iconfont.css" />
<script src="./luckysheet/plugins/js/plugin.js"></script>
<script src="./luckysheet/luckysheet.umd.js"></script>
比如这样
<script>
document.addEventListener('DOMContentLoaded', () => {
;['./luckysheet/plugins/js/plugin.js', './luckysheet/luckysheet.umd.js'].forEach((item) => {
const script = document.createElement('script')
script.defer = true
script.src = item
document.body.appendChild(script)
})
;[
'./luckysheet/plugins/css/pluginsCss.css',
'./luckysheet/plugins/plugins.css',
'./luckysheet/css/luckysheet.css',
'./luckysheet/assets/iconfont/iconfont.css'
].forEach((item) => {
const link = document.createElement('link')
link.rel = 'stylesheet'
link.type = 'text/css'
link.href = item
document.head.appendChild(link)
})
})
</script>
大小、请求数、加载时间都大大减少
注意优化后的时间chunk-vendors.js和app.js加载一共耗时1.59s,不是1.11+1.59,chunk-vendors的下载解析会阻塞app.js的下载解析,所以一共1.59s。
前:
后:
原文地址:https://juejin.cn/post/7359077652445806642
侵删
还没有使用过我们刷题网站(https://fe.ecool.fun/)或者前端面试题宝典的同学,如果近期准备或者正在找工作,千万不要错过,题库主打无广告和更新快哦~。
我们团队的前端辅导也做了将近2年了,陆陆续续辅导了几百位同学,分享一下最近几个结束辅导的回访。