Vue3?KeepAlive實現(xiàn)原理解析
思路
首先我們知道 KeepAlive 是一個內(nèi)置組件,那封裝一個組件對于大家來說應該不會有太大的困難,它的核心邏輯在于它的 render 函數(shù),它用 map 去記錄要緩存的組件,就是 [key,vnode] 的形式。它的核心思想就是 LRU,當我們限制有 maxSize 的時候,超過 maxSize 時我們會刪除最久沒有使用的[key, vnode],可以看看 leetcode146.LRU緩存;基本上你理清了 LRU 算法的思路,keepalive 的原理你也知道的差不多了。
代碼解析
我只貼出來能幫助大家理解的核心代碼,源碼位于core/package/runtime-core/src/components/KeepAlive.ts
setup
const cache: Cache = new Map()
const keys: Keys = new Set()
// cache sub tree after render
let pendingCacheKey: CacheKey | null = null
const cacheSubtree = () => {
// fix #1621, the pendingCacheKey could be 0
if (pendingCacheKey != null) {
cache.set(pendingCacheKey, getInnerChild(instance.subTree))
}
}
// 組件掛載和更新的時候就會緩存最新的組件
onMounted(cacheSubtree)
onUpdated(cacheSubtree)render
tips:這塊邏輯就是 setup 的返回值
const comp = vnode.type as ConcreteComponent
const key = vnode.key == null ? comp : vnode.key
pendingCacheKey = key
const cachedVNode = cache.get(key)
// tips: keys 僅僅是用于收集組件的 key 組件實際收集在cache中
if (cachedVNode) { // 節(jié)點存在緩存中
// copy over mounted state(復制過去的狀態(tài))
vnode.el = cachedVNode.el
vnode.component = cachedVNode.component
// 將節(jié)點放到隊尾
keys.delete(key)
keys.add(key)
} else { // 不存在
keys.add(key) // 加入緩存隊列
// prune oldest entry
// 超過最大值了,將最久沒有使用的組件刪除,也就是隊頭組件
if (max && keys.size > parseInt(max as string, 10)) {
pruneCacheEntry(keys.values().next().value)
}
}onActivated 和 onDeactived調(diào)用
tips:這兩塊邏輯是在 diff 中,看不懂沒關系,只要注意一個點就是被緩存的組件節(jié)點有一個標識 ShapeFlags.COMPONENT_SHOULD_KEEP_ALIVE,我們在掛載組件的時候和卸載組件的時候判斷一下是否有這個標識,有就去調(diào)用 onActiveated 或 onDeactived;源碼位于:core/package/runtime-core/src/renderer
if (shapeFlag & ShapeFlags.COMPONENT_SHOULD_KEEP_ALIVE) {
;(parentComponent!.ctx as KeepAliveContext).deactivate(vnode)
return
}
if (
initialVNode.shapeFlag & ShapeFlags.COMPONENT_SHOULD_KEEP_ALIVE ||
(parent &&
isAsyncWrapper(parent.vnode) &&
parent.vnode.shapeFlag & ShapeFlags.COMPONENT_SHOULD_KEEP_ALIVE)
) {
instance.a && queuePostRenderEffect(instance.a, parentSuspense)
if (
__COMPAT__ &&
isCompatEnabled(DeprecationTypes.INSTANCE_EVENT_HOOKS, instance)
) {
queuePostRenderEffect(
() => instance.emit('hook:activated'),
parentSuspense
)
}
}到此這篇關于Vue3 KeepAlive實現(xiàn)原理的文章就介紹到這了,更多相關Vue3 KeepAlive原理內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
Vue3使用Proxy實現(xiàn)數(shù)據(jù)監(jiān)聽的原因分析
在本篇文章里小編給大家整理的是一篇關于Vue3使用Proxy實現(xiàn)數(shù)據(jù)監(jiān)聽的原因分析內(nèi)容,有需要的朋友們可以跟著學習參考下。2021-11-11
使用Vue綁定class和style樣式的幾種寫法總結(jié)
這篇文章主要介紹了使用Vue綁定class和style樣式的幾種寫法,文章通過代碼示例介紹的非常詳細,具有一定的參考價值,需要的朋友可以參考下2023-07-07
Vue中對比scoped css和css module的區(qū)別
這篇文章主要介紹了Vue中scoped css和css module的區(qū)別對比,scoped css可以直接在能跑起來的vue項目中使用而css module需要增加css-loader配置才能生效。具體內(nèi)容詳情大家參考下本文2018-05-05
Vue 清除Form表單校驗信息的解決方法(清除表單驗證上次提示信息)
這篇文章主要介紹了Vue 清除Form表單校驗信息的解決方法(清除表單驗證上次提示信息),本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-04-04

