vue性能优化指南

前言

Vue框架通过数据双向绑定和虚拟DOM技术,将我们从繁琐的DOM操作中解放出来,让我们有更多的时间去思考业务逻辑;但作为一名程序员,优化问题自然是我们必须要考虑的一个问题,本文主要讲述Vue项目性能方面的优化,使项目具有更高的性能和更好的用户体验(部分内容转自:https://juejin.cn/post/6844903846385287181

1.组件的按需加载(异步加载)

当项目比较大页面比较多的时候,SPA页面在首次加载时候,就会变得很慢,这是因为vue首次加载的时候可能把看不到的页面组件也一并加载了,这个时候就需要对页面组件进行优化,使用到异步加载组件的方式。

{
    path: '/home',
    name: 'home',
    component:require('@views/home').default
}
或者下面这种
{
    path: '/home',
    name: 'home',
    component:() => import('@views/home')
}

2.屏蔽sourceMap

项目开发完成,进行打包源码上线环节,需要对项目开发环节的开发提示信息以及错误信息进行屏蔽,一方面可以减少上线代码包的大小;另一方面提高系统的安全性。在vuejs项目的config目录下有三个文件dev.env.js(开发环境配置文件)、prod.env.js(上线配置文件)、index.js(通用配置文件)。vue-cli脚手架在上线配置文件会自动设置允许sourceMap打包,所以在上线前可以屏蔽sourceMap。如下所示,index.js的配置如下,通用配置文件分别对开发环境和上线环境做了打包配置分类,在build对象中的配置信息中,productionSourceMap修改成false。

3.子组件拆分

组件拆分可以避免不必要的渲染,从而优化性能
当我们的组件的变量进行改变了之后,组件会进行重新渲染,但是很明显,有一些部分是没有必要渲染的,所以,我们可以通过把这部分的代码抽取成一个子组件来避免这些没有必要的渲染

总结就是:合理地拆分子组件可以一定程度上避免无效的渲染,减少性能的浪费

4.尽量使用computed

computed是计算属性,根据已有的变量做一些操作返回新的变量
在面对computed,watch,method的情况下,能使用computed就尽量使用computed,因为computed具有缓存的功能

5.v-for和v-if不要同时作用于同一个dom元素上

v-for 与 v-if建议不要写在同一个dom元素/组件上,不是因为会报错,而是有点浪费性能

<div v-for="item of list" v-if="showList">{{item.name}}</div>

期望:先v-if判断为true然后再进行遍历(错误)
结果:v-for的优先级高于v-if,也就是说会先进行遍历,然后再对遍历后的每个元素进行v-if判断,大大浪费性能,绝对没必要
解决办法:我们可以把v-if提到上一级元素上面,避免v-for 与 v-if写在同一个元素上

<div v-if="showList">
    <div v-for="item of list" >{{item.name}}</div>
</div>

6.v-show和v-if

v-show和v-if分不清楚?
v-if:第一次为true的时候,会创建组件,第一次为false的时候不会创建组件。切换v-if的值的时候会对组件进行卸载/重新创建
v-show:第一次无论是false还是true,都会创建组件,且不会随着v-show值的改变而卸载/重新创建。只是改变了css属性display:none,将其隐藏了起来

总结就是:
如果是切换操作比较频繁的,建议使用v-show;
如果加载了之后只会出现一种结果,建议使用v-if,特别是为false的时候直接都不用创建组件;

7.v-fo循环加唯一值key

key是为了给Vue一个提示,以便它能跟踪每个节点的身份,从而重用和重新排序现有元素,需要为每项提供一个唯一的key属性
不建议使用index作为key值,虽然在纯展示的v-for里面是不会出现问题的,但是为了养成一个好习惯,我们可以每次都使用唯一值作为key的标示。
为什么需要唯一值key那?
因为vue组件高度复用增加Key可以标识组件的唯一性,为了更好地区别各个组件 key的作用主要是为了高效的更新虚拟DOM;虚拟dom的diff比较,如果我们不添加一个唯一值key来标示这个dom,那么它们就会按照顺序来进行对比;如果添加了唯一值key来标示,自然会按照key来进行识别对比

8.keep-alive的合理使用

是Vue的内置组件,能在组件切换过程中将状态保留在内存中,会缓存不活动的组件实例,而不是销毁,防止重复渲染DOM
prop:

  • include: 字符串或正则表达式。只有匹配的组件会被缓存。
  • exclude: 字符串或正则表达式。任何匹配的组件都不会被缓存。
<keep-alive include="test-keep-alive">
  <!-- 将缓存name为test-keep-alive的组件 -->
  <component></component>
</keep-alive>

<keep-alive include="a,b">
  <!-- 将缓存name为a或者b的组件,结合动态组件使用 -->
  <component :is="view"></component>
</keep-alive>
<!-- 动态判断 -->
<keep-alive :include="includedComponents">
  <router-view></router-view>
</keep-alive>

<keep-alive exclude="test-keep-alive">
  <!-- 将不缓存name为test-keep-alive的组件 -->
  <component></component>
</keep-alive>

结合router,缓存部分页面
使用$route.meta的keepAlive属性:

<keep-alive>
    <router-view v-if="$route.meta.keepAlive"></router-view>
</keep-alive>
<router-view v-if="!$route.meta.keepAlive"></router-view>

需要在router中设置router的元信息meta:

//...router.js
export default new Router({
  routes: [
    {
      path: '/',
      name: 'Hello',
      component: Hello,
      meta: {
        keepAlive: false // 不需要缓存
      }
    },
    {
      path: '/page1',
      name: 'Page1',
      component: Page1,
      meta: {
        keepAlive: true // 需要被缓存
      }
    }
  ]
})

keep-alive生命周期钩子函数:activated、deactivated
使用<keep-alive>会将数据保留在内存中,如果要在每次进入页面的时候获取最新的数据,需要在activated阶段获取数据,承担原来created钩子中获取数据的任务

9.beforeDestory

beforeDestory这个生命周期里面做一些销毁事件的监听或者定时器的操作
不然在组件销毁的时候,有些事件的监听是不会被销毁的,需要我们主动销毁

mounted(){
    window.addEventListener('click', handleClick)
},
beforeDestory(){
    window.removeEventListener('click', handleClick)
}

10.Object.freeze()

在vue初始化的时候,会遍历data里面的数据,给data里面的数据进行响应式设置getter,setter。作用是为了响应式数据的改变然后更新视图
但是有的数据是永远不变的,用不上响应式,所以初始化设置getter,setter,就是性能的浪费。
所以可以加上Object.freeze()来表明,不做此数据的初始化。

data(){
    return {
       info:Object.freeze({
        name:'rose',
        age:18
            })
    }
}

11.内容类系统的图片资源按需加载(精灵图技术也可以使用)

对于内容类系统的图片按需加载,如果出现图片加载比较多,可以先使用v-lazy之类的懒加载库或者绑定鼠标的scroll事件,滚动到可视区域先再对数据进行加载显示,减少系统加载的数据

12.CDN的引入

  1. index.html引入
<body>
  <div id="app"></div>
  <script src="//unpkg.zhimg.com/vue@2.6.11/dist/vue.min.js"></script>
  <script src="//unpkg.zhimg.com/vue-router@3.2.0/dist/vue-router.min.js"></script>
  <script src="//unpkg.zhimg.com/vuex@3.4.0/dist/vuex.min.js"></script>
  <script src="https://cdn.bootcss.com/echarts/4.8.0/echarts.min.js"></script>
  <script src="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/10.1.1/highlight.min.js"></script>
</body>
  1. vue.config.js配置
    在vue.config.js文件中,增加externals,将引用的外部模块导入,如下:
module.exports = {
  configureWebpack: {
    externals: {
      vue: "Vue",
      "vue-router": "VueRouter",
      vuex: "Vuex",
      "echarts": "echarts",
      "highlight.js":"highlight.js"
    }
  }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,214评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,307评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,543评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,221评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,224评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,007评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,313评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,956评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,441评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,925评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,018评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,685评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,234评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,240评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,464评论 1 261
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,467评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,762评论 2 345

推荐阅读更多精彩内容