您的当前位置:首页正文

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

2021-06-24 来源:步旅网
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 获取所有的视频列表

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数组⾥⾯移除元素注意: 这两个⽅法的第⼆个参数是数组或者组件namemutations.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中匹配的

这个写法⼤家应该不会陌

⽣,这也是尤⼤神官⽅推荐的缓存⽅法, exclude属性值可以是组件名称字符串(组件选项的name属性)或者数组,代表不缓存这些组件,所以vuex⾥⾯的addExcludeComponent是代表要缓存组件,addExcludeComponent代表不缓存组件,这⾥稍微有点绕,请牢记这个规则,这样接下来你就不会被绕进去了。App.vue

接下来的两点设置⾮常重要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⾥⾯⼀次, 所以如果想追求⼏⾏代码完美解决缓存问题的,这⾥就有点⽆能为⼒了项⽬源码

项⽬源码的 (),欢迎⼤家克隆下载项⽬启动与效果演⽰

npm install安装项⽬依赖

npm run server启动后台服务器监听本地3003端⼝npm run dev启动前端项⽬三级缓存

上⾯的⽅法⼆级缓存就够了

上⾯我们说的是两个页⾯,⼆级缓存的问题,现在假设有三个页⾯,A1-A2-A3,⼀步步点进去,要求从A3返回到A2的时候,缓存A2,再从A2返回A1的时候,缓存A1,⼤家可以⾃⼰动⼿研究下,这⾥就不写了,其实就是上⾯的思路,留给⼤家研究。总结

以上就是这篇⽂章的全部内容了,希望本⽂的内容对⼤家的学习或者⼯作具有⼀定的参考学习价值,谢谢⼤家对的⽀持。

因篇幅问题不能全部显示,请点此查看更多更全内容