在开发vue的项目中有遇到了这样一个需求:一个视频列表页面,展现视频名称和是否收藏,点击进去某一项观看,能够收藏或者取消收藏,返回的时候须要记住列表页面的页码等状态,同时这条视频的收藏状态也须要更新, 可是从其余页面进来视频列表页面的时候不缓存这个页面,也就是进入的时候是视频列表页面的第一页javascript
一句话总结一下: pageAList->pageADetail->pageAList, 缓存pageAList, 同时该视频的收藏状态若是发生变化须要更新, 其余页面->pageAList, pageAList不缓存html
在网上找了不少别人的方法,都不知足咱们的需求前端
而后咱们团队几我的捣鼓了几天,还真的整出了一套方法,实现了这个需求vue
无图无真相,用一张gif图来看一下实现后的效果吧!!!java
操做流程:node
说明:ios
由于项目里面绝大部分是二级缓存,这里咱们就作二级缓存,可是这不表明个人这个缓存方法不适用三级缓存,三级缓存后面我也会讲如何实现git
用vue-cli2的脚手架搭建了一个项目,用这个项目来讲明如何实现
先来看看项目目录github
删除了无用的components目录和assets目录,新增了src/pages目录和src/store目录, pages页面用来存放页面组件, store很少说,存放vuex相关的东西,新增了server/app.js目录,用来启动后台服务vue-router
来看看服务端代码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) })
在路由配置里面把须要缓存的路由的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 } ] })
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
将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>
接下来的两点设置很是重要
对于须要缓存的一级路由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() } }
对于须要缓存的一级路由的二级路由组件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() }
进入了pageAList,就在beforeRouteEnter里缓存了它,离开当前组件的时候有两种状况:
1 跳转进去pageADetail,在pageAList的beforeRouteLeave钩子里面缓存pageAList,从pageADetail离开的时候,也有两种状况
自认为用这个方案来实现缓存,最终的效果很是完美了
缺点:
项目源码的github地址,欢迎你们克隆下载
npm install
安装项目依赖npm run server
启动后台服务器监听本地3003端口npm run dev
启动前端项目上面的方法二级缓存就够了
上面咱们说的是两个页面,二级缓存的问题,如今假设有三个页面,A1-A2-A3,一步步点进去,要求从A3返回到A2的时候,缓存A2,再从A2返回A1的时候,缓存A1,你们能够本身动手研究下,这里就不写了,其实就是上面的思路,留给你们研究,你们能够关注个人微信公众号,里面有三级缓存的代码答案。
对不起,仍是不能免俗,无论大家如何不满,我仍是要给个人公众号打广告,名字很俗,前端研究中心,可是内容不俗,不按期更新优质前端内容:原创或者翻译国外优秀教程,下面是公众号的二维码,欢迎你们扫码加入,一块儿学习和进步。