vue项目性能优化

本文介绍了Vue.js中计算属性、watch、v-for遍历优化、懒加载、服务端渲染以及Webpack配置技巧,旨在提升应用性能和开发效率。
摘要由CSDN通过智能技术生成

运用场景:

  • 当我们需要进行数值计算,并且依赖于其它数据时,应该使用 computed,因为可以利用 computed 的缓存特性,避免每次获取值时,都要重新计算;

  • 当我们需要在数据变化时执行异步或开销较大的操作时,应该使用 watch,使用 watch 选项允许我们执行异步操作 ( 访问一个 API ),限制我们执行该操作的频率,并在我们得到最终结果前,设置中间状态。这些都是计算属性无法做到的。

1.3、v-for 遍历必须为 item 添加 key,且避免同时使用 v-if

(1)v-for 遍历必须为 item 添加 key

在列表数据进行遍历渲染时,需要为每一项 item 设置唯一 key 值,方便 Vue.js 内部机制精准找到该条列表数据。当 state 更新时,新的状态值和旧的状态值对比,较快地定位到 diff 。

(2)v-for 遍历避免同时使用 v-if

v-for 比 v-if 优先级高,如果每一次都需要遍历整个数组,将会影响速度,尤其是当之需要渲染很小一部分的时候,必要情况下应该替换成 computed 属性。

推荐:

    v-for=“user in activeUsers”

    :key=“user.id”>

    {{ user.name }}

    computed: {

    activeUsers: function () {

    return this.users.filter(function (user) {

    return user.isActive

    })

    }

    }

    不推荐:

      v-for=“user in users”

      v-if=“user.isActive”

      :key=“user.id”>

      {{ user.name }}

      1.4、长列表性能优化

      Vue 会通过 Object.defineProperty 对数据进行劫持,来实现视图响应数据的变化,然而有些时候我们的组件就是纯粹的数据展示,不会有任何改变,我们就不需要 Vue 来劫持我们的数据,在大量数据展示的情况下,这能够很明显的减少组件初始化的时间,那如何禁止 Vue 劫持我们的数据呢?可以通过 Object.freeze 方法来冻结一个对象,一旦被冻结的对象就再也不能被修改了。

      export default {

      data: () => ({

      users: {}

      }),

      async created() {

      const users = await axios.get(“/api/users”);

      this.users = Object.freeze(users);

      }

      };

      1.5、事件的销毁

      Vue 组件销毁时,会自动清理它与其它实例的连接,解绑它的全部指令及事件监听器,但是仅限于组件本身的事件。如果在 js 内使用 addEventListene 等方式是不会自动销毁的,我们需要在组件销毁时手动移除这些事件的监听,以免造成内存泄露,如:

      created() {

      addEventListener(‘click’, this.click, false)

      },

      beforeDestroy() {

      removeEventListener(‘click’, this.click, false)

      }

      1.6、图片资源懒加载

      对于图片过多的页面,为了加速页面加载速度,所以很多时候我们需要将页面内未出现在可视区域内的图片先不做加载, 等到滚动到可视区域后再去加载。这样对于页面加载性能上会有很大的提升,也提高了用户体验。我们在项目中使用 Vue 的 vue-lazyload 插件:

      (1)安装插件

      npm install vue-lazyload --save-dev

      (2)在入口文件 man.js 中引入并使用

      import VueLazyload from ‘vue-lazyload’

      然后再 vue 中直接使用

      Vue.use(VueLazyload)

      或者添加自定义选项

      Vue.use(VueLazyload, {

      preLoad: 1.3,

      error: ‘dist/error.png’,

      loading: ‘dist/loading.gif’,

      attempt: 1

      })

      (3)在 vue 文件中将 img 标签的 src 属性直接改为 v-lazy ,从而将图片显示方式更改为懒加载显示:

      以上为 vue-lazyload 插件的简单使用,如果要看插件的更多参数选项,可以查看 vue-lazyload 的 github 地址。

      1.7、路由懒加载

      Vue 是单页面应用,可能会有很多的路由引入 ,这样使用 webpcak 打包后的文件很大,当进入首页时,加载的资源过多,页面会出现白屏的情况,不利于用户体验。如果我们能把不同路由对应的组件分割成不同的代码块,然后当路由被访问的时候才加载对应的组件,这样就更加高效了。这样会大大提高首屏显示的速度,但是可能其他的页面的速度就会降下来。

      路由懒加载:

      const Foo = () => import(‘./Foo.vue’)

      const router = new VueRouter({

      routes: [

      { path: ‘/foo’, component: Foo }

      ]

      })

      1.8、第三方插件的按需引入

      我们在项目中经常会需要引入第三方插件,如果我们直接引入整个插件,会导致项目的体积太大,我们可以借助 babel-plugin-component ,然后可以只引入需要的组件,以达到减小项目体积的目的。以下为项目中引入 element-ui 组件库为例:

      (1)首先,安装 babel-plugin-component :

      npm install babel-plugin-component -D

      (2)然后,将 .babelrc 修改为:

      {

      “presets”: [[“es2015”, { “modules”: false }]],

      “plugins”: [

      [

      “component”,

      {

      “libraryName”: “element-ui”,

      “styleLibraryName”: “theme-chalk”

      }

      ]

      ]

      }

      (3)在 main.js 中引入部分组件:

      import Vue from ‘vue’;

      import { Button, Select } from ‘element-ui’;

      Vue.use(Button)

      Vue.use(Select)

      1.9、优化无限列表性能

      如果你的应用存在非常长或者无限滚动的列表,那么需要采用 窗口化 的技术来优化性能,只需要渲染少部分区域的内容,减少重新渲染组件和创建 dom 节点的时间。你可以参考以下开源项目 vue-virtual-scroll-list 和 vue-virtual-scroller 来优化这种无限列表的场景的。

      1.10、服务端渲染 SSR or 预渲染

      服务端渲染是指 Vue 在客户端将标签渲染成的整个 html 片段的工作在服务端完成,服务端形成的 html 片段直接返回给客户端这个过程就叫做服务端渲染。

      (1)服务端渲染的优点:

      • 更好的 SEO:因为 SPA 页面的内容是通过 Ajax 获取,而搜索引擎爬取工具并不会等待 Ajax 异步完成后再抓取页面内容,所以在 SPA 中是抓取不到页面通过 Ajax 获取到的内容;而 SSR 是直接由服务端返回已经渲染好的页面(数据已经包含在页面中),所以搜索引擎爬取工具可以抓取渲染好的页面;

      • 更快的内容到达时间(首屏加载更快):SPA 会等待所有 Vue 编译后的 js 文件都下载完成后,才开始进行页面的渲染,文件下载等需要一定的时间等,所以首屏渲染需要一定的时间;SSR 直接由服务端渲染好页面直接返回显示,无需等待下载 js 文件及再去渲染等,所以 SSR 有更快的内容到达时间;

      (2)服务端渲染的缺点:

      • 更多的开发条件限制:例如服务端渲染只支持 beforCreate 和 created 两个钩子函数,这会导致一些外部扩展库需要特殊处理,才能在服务端渲染应用程序中运行;并且与可以部署在任何静态文件服务器上的完全静态单页面应用程序 SPA 不同,服务端渲染应用程序,需要处于 Node.js server 运行环境;

      • 更多的服务器负载:在 Node.js 中渲染完整的应用程序,显然会比仅仅提供静态文件的 server 更加大量占用CPU 资源,因此如果你预料在高流量环境下使用,请准备相应的服务器负载,并明智地采用缓存策略。

      如果你的项目的 SEO 和 首屏渲染是评价项目的关键指标,那么你的项目就需要服务端渲染来帮助你实现最佳的初始加载性能和 SEO,具体的 Vue SSR 如何实现,可以参考作者的另一篇文章《Vue SSR 踩坑之旅》。如果你的 Vue 项目只需改善少数营销页面(例如  /, /about, /contact 等)的 SEO,那么你可能需要预渲染,在构建时 (build time) 简单地生成针对特定路由的静态 HTML 文件。优点是设置预渲染更简单,并可以将你的前端作为一个完全静态的站点,具体你可以使用 prerender-spa-plugin 就可以轻松地添加预渲染 。

      二、Webpack 层面的优化


      2.1、Webpack 对图片进行压缩

      在 vue 项目中除了可以在 webpack.base.conf.js 中 url-loader 中设置 limit 大小来对图片处理,对小于 limit 的图片转化为 base64 格式,其余的不做操作。所以对有些较大的图片资源,在请求资源的时候,加载会很慢,我们可以用 image-webpack-loader来压缩图片:

      (1)首先,安装 image-webpack-loader :

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

      (2)然后,在 webpack.base.conf.js 中进行配置:

      {

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

      use:[

      {

      loader: ‘url-loader’,

      options: {

      limit: 10000,

      name: utils.assetsPath(‘img/[name].[hash:7].[ext]’)

      }

      },

      {

      loader: ‘image-webpack-loader’,

      options: {

      bypassOnDebug: true,

      }

      }

      ]

      }

      2.2、减少 ES6 转为 ES5 的冗余代码

      Babel 插件会在将 ES6 代码转换成 ES5 代码时会注入一些辅助函数,例如下面的 ES6 代码:

      class HelloWebpack extends Component{…}

      这段代码再被转换成能正常运行的 ES5 代码时需要以下两个辅助函数:

      babel-runtime/helpers/createClass // 用于实现 class 语法

      babel-runtime/helpers/inherits // 用于实现 extends 语法

      在默认情况下, Babel 会在每个输出文件中内嵌这些依赖的辅助函数代码,如果多个源代码文件都依赖这些辅助函数,那么这些辅助函数的代码将会出现很多次,造成代码冗余。为了不让这些辅助函数的代码重复出现,可以在依赖它们时通过 require('babel-runtime/helpers/createClass') 的方式导入,这样就能做到只让它们出现一次。babel-plugin-transform-runtime 插件就是用来实现这个作用的,将相关辅助函数进行替换成导入语句,从而减小 babel 编译出来的代码的文件大小。

      (1)首先,安装 babel-plugin-transform-runtime :

      npm install babel-plugin-transform-runtime --save-dev

      (2)然后,修改 .babelrc 配置文件为:

      “plugins”: [

      “transform-runtime”

      ]

      如果要看插件的更多详细内容,可以查看babel-plugin-transform-runtime 的 详细介绍。

      2.3、提取公共代码

      如果项目中没有去将每个页面的第三方库和公共模块提取出来,则项目会存在以下问题:

      • 相同的资源被重复加载,浪费用户的流量和服务器的成本。

      • 每个页面需要加载的资源太大,导致网页首屏加载缓慢,影响用户体验。

      所以我们需要将多个页面的公共代码抽离成单独的文件,来优化以上问题 。Webpack 内置了专门用于提取多个Chunk 中的公共部分的插件 CommonsChunkPlugin,我们在项目中 CommonsChunkPlugin 的配置如下:

      // 所有在 package.json 里面依赖的包,都会被打包进 vendor.js 这个文件中。

      new webpack.optimize.CommonsChunkPlugin({

      name: ‘vendor’,

      minChunks: function(module, count) {

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

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

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

      img

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

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

      如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:前端)

      最后

      技术是没有终点的,也是学不完的,最重要的是活着、不秃。零基础入门的时候看书还是看视频,我觉得成年人,何必做选择题呢,两个都要。喜欢看书就看书,喜欢看视频就看视频。最重要的是在自学的过程中,一定不要眼高手低,要实战,把学到的技术投入到项目当中,解决问题,之后进一步锤炼自己的技术。

      技术学到手后,就要开始准备面试了,找工作的时候一定要好好准备简历,毕竟简历是找工作的敲门砖,还有就是要多做面试题,复习巩固。有需要面试题资料的朋友点击这里可以领取

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

      [外链图片转存中…(img-TVZpKLQt-1712981689430)]

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

      如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:前端)

      最后

      技术是没有终点的,也是学不完的,最重要的是活着、不秃。零基础入门的时候看书还是看视频,我觉得成年人,何必做选择题呢,两个都要。喜欢看书就看书,喜欢看视频就看视频。最重要的是在自学的过程中,一定不要眼高手低,要实战,把学到的技术投入到项目当中,解决问题,之后进一步锤炼自己的技术。

      技术学到手后,就要开始准备面试了,找工作的时候一定要好好准备简历,毕竟简历是找工作的敲门砖,还有就是要多做面试题,复习巩固。有需要面试题资料的朋友点击这里可以领取

      [外链图片转存中…(img-D4pe3nEw-1712981689430)]

      评论
      添加红包

      请填写红包祝福语或标题

      红包个数最小为10个

      红包金额最低5元

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

      抵扣说明:

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

      余额充值