Vue Router路由動態(tài)緩存組件使用詳解
一、簡介
Vue Router 允許你緩存路由組件,這樣在用戶導航回之前的頁面時,組件的狀態(tài)會被保留而不是重新渲染。這可以通過使用 <keep-alive>
組件來實現(xiàn)。
<keep-alive>
是一個內(nèi)置抽象組件,它自身不會渲染一個 DOM 元素,也不會出現(xiàn)在父組件鏈中。
它的主要作用是緩存不活動的組件實例,而不是銷毀它們。
當組件在 <keep-alive>
包裹下被切換時,它的狀態(tài)(包括數(shù)據(jù)、DOM 結(jié)構(gòu)等)都會被保留,以便在下次重新顯示時能夠恢復(fù)之前的狀態(tài),而不是重新創(chuàng)建組件實例。
基本用法
<keep-alive> <component :is="currentTabComponent"></component> </keep-alive>
在上面的例子中,currentTabComponent
是一個動態(tài)組件,它的值會根據(jù)當前選中的標簽頁而變化。
使用 <keep-alive>
后,當切換標簽頁時,之前的組件實例會被緩存起來,而不是被銷毀。
生命周期鉤子
被 <keep-alive>
包裹的組件擁有兩個特有的生命周期鉤子:activated
和 deactivated
。
activated
:當組件被激活時調(diào)用,即在 keep-alive 緩存中組件再次可見時被調(diào)用。deactivated
:當組件被停用時調(diào)用,即在 keep-alive 緩存中組件不可見時被調(diào)用。
這兩個鉤子可以幫助你更好地管理組件的狀態(tài),比如當組件重新被激活時,你可能需要重新獲取數(shù)據(jù)或者更新視圖。
二、定義是否緩存組件
根據(jù)路由的 meta
字段中的 keepAlive
屬性來決定是否緩存組件。
如果 keepAlive 為 true,則組件會被緩存;否則,它會被緩存。
const routes = [ { path: '/foo', component: Foo, meta: { keepAlive: true } // 這個組件會被緩存 }, { path: '/bar', component: Bar, meta: { keepAlive: false } // 沒有設(shè)置 meta.keepAlive 或其值為 false,則組件不會被緩存 } // ... 其他路由配置 ];
三、緩存組件
1. 通過 :include 屬性實現(xiàn)
可利用keep-alive
的 include
或 exclude
屬性(include 和 exclude 包含的name 是組件的name不是router name)來設(shè)置緩存:
include
值為字符串或者正則表達式匹配的組件name會被緩存。exclude
值為字符串或正則表達式匹配的組件name不會被緩存。
vue2.x中
通過 <keep-alive>
的 :include
屬性來實現(xiàn)動態(tài)緩存組件
<template> <div id="app"> <keep-alive :include="cachedViews"> <router-view></router-view> </keep-alive> </div> </template> <script> export default { name: 'App', data() { return { cachedViews: [] // 存儲需要緩存的組件名 }; }, watch: { $route(to, from) { // 過濾出需要緩存的組件,即 meta.keepAlive = true 的組件 const keepAliveComponent = to.matched.filter(record => { return record.meta && record.meta.keepAlive; }); if (keepAliveComponent.length) { // 把需要緩存的組件的name放到cachedViews集合里 this.cachedViews = this.cachedViews.concat(keepAliveComponent.map(record => record.name)); } else { const index = this.cachedViews.indexOf(to.name); if (index > -1) { this.cachedViews.splice(index, 1); } } } } }; </script>
vue3.x中
在 Vue 3 中同樣可以使用 <keep-alive>
的 :include
屬性來實現(xiàn)動態(tài)緩存組件:
<template> <div id="app"> <keep-alive :include="cachedViews"> <router-view></router-view> </keep-alive> </div> </template> <script> import { ref, onMounted, watch, onUnmounted } from 'vue'; import { useRoute } from 'vue-router'; export default { name: 'App', setup() { const cachedViews = ref([]); const route = useRoute(); const updateCachedViews = () => { const keepAliveComponent = route.matched.filter(record => { return record.meta && record.meta.keepAlive; }); cachedViews.value = keepAliveComponent.map(record => record.name); }; onMounted(() => { updateCachedViews(); }); watch( route, () => { updateCachedViews(); }, { immediate: false } ); onUnmounted(() => { // 清理操作,如果需要的話 }); return { cachedViews }; } }; </script>
2. 通過 v-slot 功能實現(xiàn)
由于Vue 3 由于引入了組合式 API
和 v-slot
功能,有更簡潔的方式來實現(xiàn)動態(tài)緩存:
<template> <div id="app"> <router-view v-slot="{ Component }"> <keep-alive v-if="$route.meta && $route.meta.keepAlive"> <component :is="Component" /> </keep-alive> <component v-else :is="Component" /> </router-view> </div> </template> <script> import { defineComponent } from 'vue'; export default defineComponent({ name: 'App' }); </script>
使用<router-view>
的 v-slot
功能來訪問當前路由組件的實例。
然后,我們檢查這個組件的 meta.keepAlive
屬性來決定是否將其包裹在 <keep-alive>
中。
3. 通過 v-if 來實現(xiàn)
在Vue2和Vue3中都可以使用 v-if="$route.meta.keepAlive"
來判斷哪些組件需要緩存,簡單粗暴
<template> <div id="app"> <keep-alive v-if="$route.meta.keepAlive"> <router-view></router-view> </keep-alive> <router-view v-if="!$route.meta.keepAlive"></router-view> </div> </template> <script> import { defineComponent } from 'vue'; export default defineComponent({ name: 'App' }); </script>
四、注意事項
- 雖然
<keep-alive>
可以提高性能,但也要避免過度使用,因為緩存的組件實例會占用內(nèi)存。 - 當組件被
<keep-alive>
包裹時,它的created
和mounted
鉤子只會在第一次渲染時調(diào)用,之后的切換不會再次觸發(fā)這兩個鉤子。 - 如果需要強制刷新緩存的組件,可以通過改變 key 屬性或者使用 ·include/exclude· 屬性來動態(tài)控制哪些組件需要被緩存。
總結(jié)
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Vue?Router嵌套路由(children)的用法小結(jié)
嵌套路由就是父路由里面嵌套他的子路由,父路由有自己的路由導航和路由容器(router-link、router-view),通過配置children可實現(xiàn)多層嵌套,這篇文章主要介紹了Vue--Router--嵌套路由(children)的用法,需要的朋友可以參考下2022-08-08使用Vue創(chuàng)建前后端分離項目的完整過程(前端部分)
這篇文章主要介紹了使用Vue.js和Node.js搭建一個前端和后端分離的項目,并使用VueCLI3腳手架、axios進行HTTP請求、Vue-router實現(xiàn)前端路由和vuex進行狀態(tài)管理,需要的朋友可以參考下2025-01-01