vue项目中实现缓存的最佳方案详解


Posted in Javascript onJuly 11, 2019

需求

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

vue项目中实现缓存的最佳方案详解

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

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

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

实现后的效果

无图无真相,用一张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,添加两个路由生命周期钩子beforeRouteEnter和beforeRouteLeave

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里面一次, 所以如果想追求几行代码完美解决缓存问题的,这里就有点无能为力了

项目源码

项目源码的github地址 (本地下载),欢迎大家克隆下载

项目启动与效果演示

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

三级缓存

上面的方法二级缓存就够了

上面我们说的是两个页面,二级缓存的问题,现在假设有三个页面,A1-A2-A3,一步步点进去,要求从A3返回到A2的时候,缓存A2,再从A2返回A1的时候,缓存A1,大家可以自己动手研究下,这里就不写了,其实就是上面的思路,留给大家研究。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对三水点靠木的支持。

Javascript 相关文章推荐
文本框根据输入内容自适应高度的代码
Oct 24 Javascript
节点的插入之append()和appendTo()的用法介绍
Jan 13 Javascript
jquery.map()方法的使用详解
Jul 09 Javascript
jquery+php随机生成红包金额数量代码分享
Aug 27 Javascript
jQuery实现带有动画效果的回到顶部和底部代码
Nov 04 Javascript
js实现浏览器倒计时跳转页面效果
Aug 12 Javascript
关于vue的语法规则检测报错问题的解决
May 21 Javascript
react配置antd按需加载的使用
Feb 11 Javascript
vue项目强制清除页面缓存的例子
Nov 06 Javascript
jQuery实现简单弹幕效果
Nov 28 jQuery
Vue和React有哪些区别
Sep 12 Javascript
react中hook介绍以及使用教程
Dec 11 Javascript
浅入深出Vue之组件使用
Jul 11 #Javascript
elementUI 动态生成几行几列的方法示例
Jul 11 #Javascript
JS为什么说async/await是generator的语法糖详解
Jul 11 #Javascript
Vue 使用计时器实现跑马灯效果的实例代码
Jul 11 #Javascript
Vue 子组件与数据传递问题及注意事项
Jul 11 #Javascript
ES6的异步终极解决方案分享
Jul 11 #Javascript
微信小程序的开发范式BeautyWe.js入门详解
Jul 10 #Javascript
You might like
一步一步学习PHP(5) 类和对象
2010/02/16 PHP
php curl post 时出现的问题解决
2014/01/30 PHP
PHP跨平台获取服务器IP地址自定义函数分享
2014/12/29 PHP
自己写的php中文截取函数mb_strlen和mb_substr
2015/02/09 PHP
PHP实现基于mysqli的Model基类完整实例
2016/04/08 PHP
PHP设计模式之单例模式原理与实现方法分析
2018/04/25 PHP
PHP连接SQL server数据库测试脚本运行实例
2020/08/24 PHP
使用jquery菜单插件HoverTree仿京东无限级菜单
2014/12/18 Javascript
javascript运算符——位运算符全面介绍
2016/07/14 Javascript
AngularJS基于ngInfiniteScroll实现下拉滚动加载的方法
2016/12/14 Javascript
微信小程序 登陆流程详细介绍
2017/01/17 Javascript
Bootstrap组合上、下拉框简单实现代码
2017/03/06 Javascript
vue.js 1.x与2.0中js实时监听input值的变化
2017/03/15 Javascript
微信小程序搜索组件wxSearch实例详解
2017/06/08 Javascript
vue 实现axios拦截、页面跳转和token 验证
2018/07/17 Javascript
JavaScript常用工具方法封装
2019/02/12 Javascript
使用weixin-java-miniapp配置进行单个小程序的配置详解
2019/03/29 Javascript
vue中使用v-model完成组件间的通信
2019/08/22 Javascript
[08:40]Navi Vs Newbee
2018/06/07 DOTA
django如何连接已存在数据的数据库
2018/08/14 Python
python开发准备工作之配置虚拟环境(非常重要)
2019/02/11 Python
Python 从subprocess运行的子进程中实时获取输出的例子
2019/08/14 Python
python 内置函数汇总详解
2019/09/16 Python
使用OpenCV circle函数图像上画圆的示例代码
2019/12/27 Python
阿里巴巴国际站:Alibaba.com
2016/07/21 全球购物
开会迟到检讨书
2014/01/08 职场文书
工厂总经理岗位职责
2014/02/07 职场文书
社会工作专业求职信
2014/07/15 职场文书
小学班级特色活动方案
2014/08/31 职场文书
2014年销售经理工作总结
2014/12/01 职场文书
小学教师个人总结
2015/02/05 职场文书
物业前台接待岗位职责
2015/04/03 职场文书
手机销售员岗位职责
2015/04/11 职场文书
出生证明范本
2015/06/15 职场文书
2016党员学习作风建设心得体会
2016/01/21 职场文书
动画电影《龙珠超 超级英雄》延期上映
2022/03/20 日漫