vue項(xiàng)目中實(shí)現(xiàn)緩存的最佳方案詳解
需求
在開發(fā)vue的項(xiàng)目中有遇到了這樣一個(gè)需求:一個(gè)視頻列表頁面,展示視頻名稱和是否收藏,點(diǎn)擊進(jìn)去某一項(xiàng)觀看,可以收藏或者取消收藏,返回的時(shí)候需要記住列表頁面的頁碼等狀態(tài),同時(shí)這條視頻的收藏狀態(tài)也需要更新, 但是從其他頁面進(jìn)來視頻列表頁面的時(shí)候不緩存這個(gè)頁面,也就是進(jìn)入的時(shí)候是視頻列表頁面的第一頁
一句話總結(jié)一下: pageAList->pageADetail->pageAList, 緩存pageAList, 同時(shí)該視頻的收藏狀態(tài)如果發(fā)生變化需要更新, 其他頁面->pageAList, pageAList不緩存
在網(wǎng)上找了很多別人的方法,都不滿足我們的需求
然后我們團(tuán)隊(duì)幾個(gè)人搗鼓了幾天,還真的整出了一套方法,實(shí)現(xiàn)了這個(gè)需求
實(shí)現(xiàn)后的效果
無圖無真相,用一張gif圖來看一下實(shí)現(xiàn)后的效果吧?。?!
操作流程:
- 首頁->pageAList, 跳轉(zhuǎn)第二頁 ->首頁-> pageAList,頁碼顯示第一頁,說明從其他頁面進(jìn)入pageAList, pageAList頁面沒有被緩存
- pageAList, 跳轉(zhuǎn)到第三頁,點(diǎn)擊視頻22 -> 進(jìn)入視頻詳情頁pageADetail,點(diǎn)擊收藏,收藏成功,點(diǎn)擊返回 -> pageAList顯示的是第三頁,并且視頻22的收藏狀態(tài)從未收藏變成已收藏,說明從pageADetail進(jìn)入pageAList,pageAList頁面緩存了,并且更新了狀態(tài)
說明:
- 二級緩存: 也就是從A->B->A,緩存A
- 三級緩存:A->B->C->B->A, 緩存A,B 因?yàn)轫?xiàng)目里面絕大部分是二級緩存,這里我們就做二級緩存,但是這不代表我的這個(gè)緩存方法不適用三級緩存,三級緩存后面我也會(huì)講如何實(shí)現(xiàn)
實(shí)現(xiàn)二級緩存
用vue-cli2的腳手架搭建了一個(gè)項(xiàng)目,用這個(gè)項(xiàng)目來說明如何實(shí)現(xiàn)
先來看看項(xiàng)目目錄
刪除了無用的components目錄和assets目錄,新增了src/pages目錄和src/store目錄, pages頁面用來存放頁面組件, store不多說,存放vuex相關(guān)的東西,新增了server/app.js目錄,用來啟動(dòng)后臺(tái)服務(wù)
1. 前提條件
- 項(xiàng)目引入vue,vuex, vue-router,axios等vue全家桶
- 引入element-ui,只是為了項(xiàng)目美觀,畢竟本人懶癌晚期,不想自己寫樣式
- 在config/index.js里面配置前端代理
- 引入express,啟動(dòng)后臺(tái),后端開3003端口,給前端提供api支持 來看看服務(wù)端代碼server/app.js,非常簡單,就是造了30條數(shù)據(jù),寫了3個(gè)接口,幾十行文件直接搭建了一個(gè)node服務(wù)器,簡單粗暴解決數(shù)據(jù)模擬問題,會(huì)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 })) // 后臺(tái)設(shè)置允許跨域訪問 // 前后端都是本地localhost,所以不需要設(shè)置cors跨域,如果是部署在服務(wù)器上,則需要設(shè)置 // 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, 這個(gè)想必大家都知道,是緩存路由組件的
在我們項(xiàng)目里面,需要緩存的路由是pageAList,所以這個(gè)路由的meta的keepAlive設(shè)置成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里面存儲(chǔ)一個(gè)名為excludeComponents的數(shù)組,這個(gè)數(shù)組用來操作需要做緩存的組件
state.js
const state = { excludeComponents: [] } export default state
同時(shí)在mutations.js里面加入兩個(gè)方法, addExcludeComponent是往excludeComponents里面添加元素的,removeExcludeComponent是往excludeComponents數(shù)組里面移除元素
注意: 這兩個(gè)方法的第二個(gè)參數(shù)是數(shù)組或者組件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字符串或者數(shù)組 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的路由也需要這么包裹,這點(diǎn)非常重要,因?yàn)閜ageAList組件是從它們的router-view中匹配的
<keep-alive :exclude="excludeComponents"><som-component></some-component></keep-alive>這個(gè)寫法大家應(yīng)該不會(huì)陌生,這也是尤大神官方推薦的緩存方法, exclude屬性值可以是組件名稱字符串(組件選項(xiàng)的name屬性)或者數(shù)組,代表不緩存這些組件,所以vuex里面的addExcludeComponent是代表要緩存組件,addExcludeComponent代表不緩存組件,這里稍微有點(diǎn)繞,請牢記這個(gè)規(guī)則,這樣接下來你就不會(huì)被繞進(jìn)去了。
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>
接下來的兩點(diǎn)設(shè)置非常重要
5. 一級組件
對于需要緩存的一級路由pageAList,添加兩個(gè)路由生命周期鉤子beforeRouteEnter和beforeRouteLeave
import {getVideoList} from '../api' export default { name: 'pageAList', // 組件名稱,和組件對應(yīng)的路由名稱不需要相同 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,進(jìn)入這個(gè)組件pageAList之前,在excludeComponents移除當(dāng)前組件,也就是緩存當(dāng)前組件,所以任何路由跳轉(zhuǎn)到這個(gè)組件,這個(gè)組件其實(shí)都是被緩存的,都會(huì)觸發(fā)activated鉤子
- beforeRouteLeave: 離開當(dāng)前頁面,如果跳轉(zhuǎn)到pageADetail,那么就需要在excludeComponents移除當(dāng)前組件pageAList,也就是緩存當(dāng)前組件,如果是跳轉(zhuǎn)到其他頁面,就需要把pageAList添加進(jìn)去excludeComponents,也就是不緩存當(dāng)前組件
- 獲取數(shù)據(jù)的方法getVideoList在mounted或者created鉤子里面調(diào)取,如果二級路由更改數(shù)據(jù),一級路由需要更新,那么就需要在activated鉤子里再獲取一次數(shù)據(jù),我們這個(gè)詳情可以收藏,改變列表的狀態(tài),所以這兩個(gè)鉤子都使用了
6. 二級組件
對于需要緩存的一級路由的二級路由組件pageADetail,添加beforeRouteLeave路由生命周期鉤子
在這個(gè)beforeRouteLeave鉤子里面,需要先清除一級組件的緩存狀態(tài),如果跳轉(zhuǎn)路由匹配到一級組件,再緩存一級組件
beforeRouteLeave (to, from, next) { let componentName = '' // 離開詳情頁時(shí),將pageAList添加到exludeComponents里,也就是將需要緩存的頁面pageAList置為不緩存狀態(tài) let list = ['pageAList'] this.$store.commit('addExcludeComponent', list) // 緩存組件路由名稱到組件name的映射 let map = new Map([['pageAList', 'pageAList']]) componentName = map.get(to.name) || '' // 如果離開的時(shí)候跳轉(zhuǎn)的路由是pageAList,將pageAList從exludeComponents里面移除,也就是要緩存pageAList this.$store.commit('removeExcludeComponent', componentName) next() }
7.實(shí)現(xiàn)方法總結(jié)
進(jìn)入了pageAList,就在beforeRouteEnter里緩存了它,離開當(dāng)前組件的時(shí)候有兩種情況:
1 跳轉(zhuǎn)進(jìn)去pageADetail,在pageAList的beforeRouteLeave鉤子里面緩存pageAList,從pageADetail離開的時(shí)候,也有兩種情況
(1) 回到pageAList,那么在pageADetail的beforeRouteLeave鉤子里面緩存了pageAList,所以這就是從pageAList-pageADetail-pageAList的時(shí)候,pageAList可以被緩存,還是之前的頁碼狀態(tài)
(2) 進(jìn)入其他路由,在pageADetail的beforeRouteLeave鉤子里面清除了pageAList的緩存
2 跳轉(zhuǎn)到非pageADetail的頁面,在pageAList的beforeRouteLeave鉤子里面清除pageAList的緩存
方案評估
自認(rèn)為用這個(gè)方案來實(shí)現(xiàn)緩存,最終的效果非常完美了
缺點(diǎn):
- 代碼有點(diǎn)多,緩存代碼不好復(fù)用
- 性能問題:如果在要緩存的一級組件里面寫了activated鉤子,那么從非一級組件對應(yīng)的二級組件進(jìn)入到要緩存的一級組件的時(shí)候,會(huì)發(fā)送兩次接口請求數(shù)據(jù),mounted里面一次, activated里面一次, 所以如果想追求幾行代碼完美解決緩存問題的,這里就有點(diǎn)無能為力了
項(xiàng)目源碼
項(xiàng)目源碼的github地址 (本地下載),歡迎大家克隆下載
項(xiàng)目啟動(dòng)與效果演示
- npm install安裝項(xiàng)目依賴
- npm run server啟動(dòng)后臺(tái)服務(wù)器監(jiān)聽本地3003端口
- npm run dev啟動(dòng)前端項(xiàng)目
三級緩存
上面的方法二級緩存就夠了
上面我們說的是兩個(gè)頁面,二級緩存的問題,現(xiàn)在假設(shè)有三個(gè)頁面,A1-A2-A3,一步步點(diǎn)進(jìn)去,要求從A3返回到A2的時(shí)候,緩存A2,再從A2返回A1的時(shí)候,緩存A1,大家可以自己動(dòng)手研究下,這里就不寫了,其實(shí)就是上面的思路,留給大家研究。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對腳本之家的支持。
相關(guān)文章
Vue3實(shí)現(xiàn)轉(zhuǎn)盤抽獎(jiǎng)效果的示例詳解
這篇文章主要為大家詳細(xì)介紹了如何通過Vue3實(shí)現(xiàn)簡單的轉(zhuǎn)盤抽獎(jiǎng)效果,文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,感興趣的可以了解一下2023-10-10Vue關(guān)于數(shù)據(jù)綁定出錯(cuò)解決辦法
這篇文章主要介紹了Vue關(guān)于數(shù)據(jù)綁定出錯(cuò)解決辦法的相關(guān)資料,需要的朋友可以參考下2017-05-05vue路由傳參的基本實(shí)現(xiàn)方式小結(jié)【三種方式】
這篇文章主要介紹了vue路由傳參的基本實(shí)現(xiàn)方式,結(jié)合實(shí)例形式總結(jié)分析了vue.js路由傳參的三種實(shí)現(xiàn)方式,包括params顯示傳參、不顯示參數(shù)以及query顯示參數(shù)傳參,需要的朋友可以參考下2020-02-02swiper在vue項(xiàng)目中l(wèi)oop循環(huán)輪播失效的解決方法
今天小編就為大家分享一篇swiper在vue項(xiàng)目中l(wèi)oop循環(huán)輪播失效的解決方法,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-09-09Vue.JS實(shí)現(xiàn)垂直方向展開、收縮不定高度模塊的JS組件
這篇文章主要介紹了Vue.JS實(shí)現(xiàn)垂直方向展開、收縮不定高度模塊的JS組件,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2018-06-06element日期組件實(shí)現(xiàn)只能選擇小時(shí)或分鐘
本文主要介紹了element日期組件實(shí)現(xiàn)只能選擇小時(shí)或分鐘,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-01-01