vue项⽬中实现缓存的最佳⽅案详解
需求
在开发vue的项⽬中有遇到了这样⼀个需求:⼀个视频列表页⾯,展⽰视频名称和是否收藏,点击进去某⼀项观看,可以收藏或者取消收藏,返回的时候需要记住列表页⾯的页码等状态,同时这条视频的收藏状态也需要更新, 但是从其他页⾯进来视频列表页⾯的时候不缓存这个页⾯,也就是进⼊的时候是视频列表页⾯的第⼀页
⼀句话总结⼀下: pageAList->pageADetail->pageAList, 缓存pageAList, 同时该视频的收藏状态如果发⽣变化需要更新,其他页⾯->pageAList, pageAList不缓存
后端字符串转数组在⽹上了很多别⼈的⽅法,都不满⾜我们的需求
然后我们团队⼏个⼈捣⿎了⼏天,还真的整出了⼀套⽅法,实现了这个需求
实现后的效果
⽆图⽆真相,⽤⼀张gif图来看⼀下实现后的效果吧
操作流程:
⾸页->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的脚⼿架搭建了⼀个项⽬,⽤这个项⽬来说明如何实现
先来看看项⽬⽬录
删除了⽆⽤的components⽬录和assets⽬录,新增了src/pages⽬录和src/store⽬录, pages页⾯⽤来存放页⾯组件, store不多说,存放vuex相关的东西,新增了server/app.js⽬录,⽤来启动后台服务
1. 前提条件
项⽬引⼊vue,vuex, vue-router,axios等vue全家桶
引⼊element-ui,只是为了项⽬美观,毕竟本⼈懒癌晚期,不想⾃⼰写样式
在config/index.js⾥⾯配置前端代理
引⼊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 获取所有的视频列表
<('/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 获取某⼀条视频详情
<('/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 = ludeComponents
if (Array.isArray(excludeComponent)) {
} else {
}
},
// excludeComponent可能是组件name字符串或者数组
removeExcludeComponent (state, excludeComponent) {
let excludeComponents = ludeComponents
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
}
}
}
}
}
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="$a.keepAlive"></router-view>
</keep-alive>
<router-view v-if="!$a.keepAlive"></router-view>
</div>
</template>
<script>
export default {
name: 'App',
computed: {
excludeComponents () {
return this.$ludeComponents
}
}
}
</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="$a.keepAlive"></router-view>
</keep-alive>
<router-view v-if="!$a.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.$ludeComponents
}
},
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
}
})
},
goIntoVideo (item) {
this.$router.push({name: 'pageADetail', query: {id: item.id}})
},
handleCurrentPage (val) {
this.currentPage = val
}
},
beforeRouteEnter (to, from, next) {
next(vm => {
vm.$storemit('removeExcludeComponent', 'pageAList')
next()
})
},
beforeRouteLeave (to, from, next) {
let reg = /pageADetail/
if (st(to.name)) {
this.$storemit('removeExcludeComponent', 'pageAList')
} else {
this.$storemit('addExcludeComponent', 'pageAList')
}
next()
},
activated () {
},
mounted () {
}
}
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.$storemit('addExcludeComponent', list)
// 缓存组件路由名称到组件name的映射
let map = new Map([['pageAList', 'pageAList']])
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论