欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

vue项目缓存最佳实践

程序员文章站 2022-04-30 20:28:37
需求 在开发vue的项目中有遇到了这样一个需求:一个视频列表页面,展示视频名称和是否收藏,点击进去某一项观看,可以收藏或者取消收藏,返回的时候需要记住列表页面的页码等状态,同时这条视频的收藏状态也需要更新, 但是从其他页面进来视频列表页面的时候不缓存这个页面,也就是进入的时候是视频列表页面的第一页 ......

需求

在开发vue的项目中有遇到了这样一个需求:一个视频列表页面,展示视频名称和是否收藏,点击进去某一项观看,可以收藏或者取消收藏,返回的时候需要记住列表页面的页码等状态,同时这条视频的收藏状态也需要更新, 但是从其他页面进来视频列表页面的时候不缓存这个页面,也就是进入的时候是视频列表页面的第一页

vue项目缓存最佳实践

一句话总结一下: pagealist->pageadetail->pagealist, 缓存pagealist, 同时该视频的收藏状态如果发生变化需要更新, 其他页面->pagealist, pagealist不缓存

在网上找了很多别人的方法,都不满足我们的需求

vue项目缓存最佳实践

然后我们团队几个人捣鼓了几天,还真的整出了一套方法,实现了这个需求

vue项目缓存最佳实践

实现后的效果

无图无真相,用一张gif图来看一下实现后的效果吧!!!

vue项目缓存最佳实践

操作流程:

  • 首页->pagealist, 跳转第二页 ->首页-> pagealist,页码显示第一页,说明从其他页面进入pagealist, pagealist页面没有被缓存
  • pagealist, 跳转到第三页,点击视频22 -> 进入视频详情页pageadetail,点击收藏,收藏成功,点击返回 -> pagealist显示的是第三页,并且视频22的收藏状态从未收藏变成已收藏,说明从pageadetail进入pagealist,pagealist页面缓存了,并且更新了状态

说明:

  • 二级缓存: 也就是从a->b->a,缓存a
  • 三级缓存:a->b->c->b->a, 缓存a,b
    因为项目里面绝大部分是二级缓存,这里我们就做二级缓存,但是这不代表我的这个缓存方法不适用三级缓存,三级缓存后面我也会讲如何实现

实现二级缓存

用vue-cli2的脚手架搭建了一个项目,用这个项目来说明如何实现
先来看看项目目录

vue项目缓存最佳实践
删除了无用的components目录和assets目录,新增了src/pages目录和src/store目录, pages页面用来存放页面组件, store不多说,存放vuex相关的东西,新增了server/app.js目录,用来启动后台服务

 

1. 前提条件

  • 项目引入vue,vuex, vue-router,axios等vue全家桶
  • 引入element-ui,只是为了项目美观,毕竟本人懒癌晚期,不想自己写样式
  • 在config/index.js里面配置前端代理
    vue项目缓存最佳实践
  • 引入express,启动后台,后端开3003端口,给前端提供api支持
    来看看服务端代码server/app.js,非常简单,就是造了30条数据,写了3个接口,几十行文件直接搭建了一个node服务器,简单粗暴解决数据模拟问题,会mock用mock也行
const express = require('express')
// const bodyparser = require('body-parser')
const app = express()
let alllist = array.from({length: 30}, (v, i) => ({
  id: i,
  name: '视频' + i,
  iscollect: false
}))
// 后台设置允许跨域访问
// 前后端都是本地localhost,所以不需要设置cors跨域,如果是部署在服务器上,则需要设置
// app.all('*', function (req, res, next) {
//   res.header('access-control-allow-origin', '*')
//   res.header('access-control-allow-headers', 'x-requested-with')
//   res.header('access-control-allow-methods', 'put,post,get,delete,options')
//   res.header('x-powered-by', ' 3.2.1')
//   res.header('content-type', 'application/json;charset=utf-8')
//   next()
// })
app.use(express.json())
app.use(express.urlencoded({extended: false}))
// 1 获取所有的视频列表
app.get('/api/getvideolist', function (req, res) {
  let query = req.query
  let currentpage = query.currentpage
  let pagesize = query.pagesize
  let list = alllist.slice((currentpage - 1) * pagesize, currentpage * pagesize)
  res.json({
    code: 0,
    data: {
      list,
      total: alllist.length
    }
  })
})
// 2 获取某一条视频详情
app.get('/api/getvideodetail/:id', function (req, res) {
  let id = number(req.params.id)
  let info = alllist.find(v => v.id === id)
  res.json({
    code: 0,
    data: info
  })
})
// 3 收藏或者取消收藏视频
app.post('/api/collectvideo', function (req, res) {
  let id = number(req.body.id)
  let iscollect = req.body.iscollect
  alllist = alllist.map((v, i) => {
    return v.id === id ? {...v, iscollect} : v
  })
  res.json({code: 0})
})
const port = 3003
app.listen(port, function () {
  console.log('app is listening port' + port)
})

2. 路由配置

在路由配置里面把需要缓存的路由的meta添加keepalive属性,值为true, 这个想必大家都知道,是缓存路由组件的
在我们项目里面,需要缓存的路由是pagealist,所以这个路由的meta的keepalive设置成true,其他路由正常写,路由文件src/router/index.js如下:

import vue from 'vue'
import router from 'vue-router'
import home from '../pages/home'
import pagealist from '../pages/pagealist'
import pageadetail from '../pages/pageadetail'
import pageb from '../pages/pageb'
import main from '../pages/main'
vue.use(router)

export default new router({
  routes: [
    {
      path: '/',
      name: 'main',
      component: main,
      redirect: '/home',
      children: [
        {
          path: 'home',
          name: 'home',
          component: home
        },
        {
          path: 'pagealist',
          name: 'pagealist',
          component: pagealist,
          meta: {
            keepalive: true
          }
        },
        {
          path: 'pageb',
          component: pageb
        }
      ]
    },
    {
      path: '/pageadetail',
      name: 'pageadetail',
      component: pageadetail
    }
  ]
})

3. vuex配置

vuex的store.js里面存储一个名为excludecomponents的数组,这个数组用来操作需要做缓存的组件

state.js

const state = {
  excludecomponents: [] 
}
export default state

同时在mutations.js里面加入两个方法, addexcludecomponent是往excludecomponents里面添加元素的,removeexcludecomponent是往excludecomponents数组里面移除元素

注意: 这两个方法的第二个参数是数组或者组件name

mutations.js

const mutations = {
  addexcludecomponent (state, excludecomponent) {
    let excludecomponents = state.excludecomponents
    if (array.isarray(excludecomponent)) {
      state.excludecomponents = [...new set([...excludecomponents, ...excludecomponent])]
    } else {
      state.excludecomponents = [...new set([...excludecomponents, excludecomponent])]
    }
  },
  // excludecomponent可能是组件name字符串或者数组
  removeexcludecomponent (state, excludecomponent) {
    let excludecomponents = state.excludecomponents
    if (array.isarray(excludecomponent)) {
      for (let i = 0; i < excludecomponent.length; i++) {
        let index = excludecomponents.findindex(v => v === excludecomponent[i])
        if (index > -1) {
          excludecomponents.splice(index, 1)
        }
      }
    } else {
      for (let i = 0, len = excludecomponents.length; i < len; i++) {
        if (excludecomponents[i] === excludecomponent) {
          excludecomponents.splice(i, 1)
          break
        }
      }
    }
    state.excludecomponents = excludecomponents
  }
}
export default mutations

4. keep-alive包裹router-view

将app.vue的router-view用keep-alive组件包裹, main.vue的路由也需要这么包裹,这点非常重要,因为pagealist组件是从它们的router-view中匹配的

<keep-alive :exclude="excludecomponents"><som-component></some-component></keep-alive>这个写法大家应该不会陌生,这也是尤大神官方推荐的缓存方法, exclude属性值可以是组件名称字符串(组件选项的name属性)或者数组,代表不缓存这些组件,所以vuex里面的addexcludecomponent是代表要缓存组件,addexcludecomponent代表不缓存组件,这里稍微有点绕,请牢记这个规则,这样接下来你就不会被绕进去了。

app.vue

<template>
  <div id="app">
    <keep-alive :exclude="excludecomponents">
      <router-view v-if="$route.meta.keepalive"></router-view>
    </keep-alive>
    <router-view v-if="!$route.meta.keepalive"></router-view>
  </div>
</template>

<script>
export default {
  name: 'app',
  computed: {
    excludecomponents () {
      return this.$store.state.excludecomponents
    }
  }
}
</script

main.vue

<template>
  <div>
    <ul>
      <li v-for="nav in navs" :key="nav.name">
        <router-link :to="nav.name">{{nav.title}}</router-link>
      </li>
    </ul>
    <keep-alive :exclude="excludecomponents">
      <router-view v-if="$route.meta.keepalive"></router-view>
    </keep-alive>
    <router-view v-if="!$route.meta.keepalive"></router-view>
  </div>
</template>
<script>
export default {
  name: 'main.vue',
  data () {
    return {
      navs: [{
        name: 'home',
        title: '首页'
      }, {
        name: 'pagealist',
        title: 'pagealist'
      }, {
        name: 'pageb',
        title: 'pageb'
      }]
    }
  },
  methods: {
  },
  computed: {
    excludecomponents () {
      return this.$store.state.excludecomponents
    }
  },
  created () {
  }
}
</script>

接下来的两点设置非常重要

5. 一级组件

对于需要缓存的一级路由pagealist,添加两个路由生命周期钩子beforerouteenterbeforerouteleave

import {getvideolist} from '../api'
export default {
  name: 'pagealist', // 组件名称,和组件对应的路由名称不需要相同
  data () {
    return {
      currentpage: 1,
      pagesize: 10,
      total: 0,
      alllist: [],
      list: []
    }
  },
  methods: {
    getvideolist () {
      let params = {currentpage: this.currentpage, pagesize: this.pagesize}
      getvideolist(params).then(r => {
        if (r.code === 0) {
          this.list = r.data.list
          this.total = r.data.total
        }
      })
    },
    gointovideo (item) {
      this.$router.push({name: 'pageadetail', query: {id: item.id}})
    },
    handlecurrentpage (val) {
      this.currentpage = val
      this.getvideolist()
    }
  },
  beforerouteenter (to, from, next) {
    next(vm => {
      vm.$store.commit('removeexcludecomponent', 'pagealist')
      next()
    })
    },
  beforerouteleave (to, from, next) {
    let reg = /pageadetail/
    if (reg.test(to.name)) {
      this.$store.commit('removeexcludecomponent', 'pagealist')
    } else {
      this.$store.commit('addexcludecomponent', 'pagealist')
    }
    next()
  },
  activated () {
    this.getvideolist()
  },
  mounted () {
    this.getvideolist()
  }
}
  • beforerouteenter,进入这个组件pagealist之前,在excludecomponents移除当前组件,也就是缓存当前组件,所以任何路由跳转到这个组件,这个组件其实都是被缓存的,都会触发activated钩子
  • beforerouteleave: 离开当前页面,如果跳转到pageadetail,那么就需要在excludecomponents移除当前组件pagealist,也就是缓存当前组件,如果是跳转到其他页面,就需要把pagealist添加进去excludecomponents,也就是不缓存当前组件
  • 获取数据的方法getvideolist在mounted或者created钩子里面调取,如果二级路由更改数据,一级路由需要更新,那么就需要在activated钩子里再获取一次数据,我们这个详情可以收藏,改变列表的状态,所以这两个钩子都使用了

6. 二级组件

对于需要缓存的一级路由的二级路由组件pageadetail,添加beforerouteleave路由生命周期钩子

在这个beforerouteleave钩子里面,需要先清除一级组件的缓存状态,如果跳转路由匹配到一级组件,再缓存一级组件

beforerouteleave (to, from, next) {
    let componentname = ''
    // 离开详情页时,将pagealist添加到exludecomponents里,也就是将需要缓存的页面pagealist置为不缓存状态
    let list = ['pagealist']
    this.$store.commit('addexcludecomponent', list)
    // 缓存组件路由名称到组件name的映射
    let map = new map([['pagealist', 'pagealist']])
    componentname = map.get(to.name) || ''
    // 如果离开的时候跳转的路由是pagealist,将pagealist从exludecomponents里面移除,也就是要缓存pagealist
    this.$store.commit('removeexcludecomponent', componentname)
    next()
  }

7.实现方法总结

  • 进入了pagealist,就在beforerouteenter里缓存了它,离开当前组件的时候有两种情况:
    • 1 跳转进去pageadetail,在pagealist的beforerouteleave钩子里面缓存pagealist,从pageadetail离开的时候,也有两种情况
      • (1) 回到pagealist,那么在pageadetail的beforerouteleave钩子里面缓存了pagealist,所以这就是从pagealist-pageadetail-pagealist的时候,pagealist可以被缓存,还是之前的页码状态
      • (2) 进入其他路由,在pageadetail的beforerouteleave钩子里面清除了pagealist的缓存
    • 2 跳转到非pageadetail的页面,在pagealist的beforerouteleave钩子里面清除pagealist的缓存

方案评估

自认为用这个方案来实现缓存,最终的效果非常完美了
缺点:

  1. 代码有点多,缓存代码不好复用
  2. 性能问题:如果在要缓存的一级组件里面写了activated钩子,那么从非一级组件对应的二级组件进入到要缓存的一级组件的时候,会发送两次接口请求数据,mounted里面一次, activated里面一次, 所以如果想追求几行代码完美解决缓存问题的,这里就有点无能为力了

项目源码

项目源码的,欢迎大家克隆下载

项目启动与效果演示

  1. npm install安装项目依赖
  2. npm run server启动后台服务器监听本地3003端口
  3. npm run dev启动前端项目

三级缓存

上面的方法二级缓存就够了
上面我们说的是两个页面,二级缓存的问题,现在假设有三个页面,a1-a2-a3,一步步点进去,要求从a3返回到a2的时候,缓存a2,再从a2返回a1的时候,缓存a1,大家可以自己动手研究下,这里就不写了,其实就是上面的思路,留给大家研究,大家可以关注我的微信公众号,里面有三级缓存的代码答案。

对不起,还是不能免俗,不管你们如何不满,我还是要给我的公众号打广告,名字很俗,前端研究中心,但是内容不俗,不定期更新优质前端内容:原创或者翻译国外优秀教程,下面是公众号的二维码,欢迎大家扫码加入,一起学习和进步。

vue项目缓存最佳实践

近期优质文章