vue2源码解析之keep-alive
基本使用
Props:
include
- 字符串或正则表达式。只有名称匹配的组件会被缓存。exclude
- 字符串或正则表达式。任何名称匹配的组件都不会被缓存。max
- 数字。最多可以缓存多少组件实例。用法:
<keep-alive>
包裹动态组件时,会缓存不活动的组件实例,而不是销毁它们。 和<transition>
相似,<keep-alive>
是一个抽象组件:它自身不会渲染 一个 DOM 元素,也不会出现在组件的父组件链中。 当组件在<keep-alive>
内被切换,它的activated
和deactivated
这两个生命周期钩子函数将会被对应执行。注意,
<keep-alive>
是用在其一个直属的子组件被开关的情形。如果你在其 中有v-for
则不会工作。如果有上述的多个条件性的子元素,<keep-alive>
要求同时只有一个子元素被渲染。
include
和exclude
prop 允许组件有条件地缓存。二者都可以用逗号分 隔字符串、正则表达式或一个数组来表示:匹配首先检查组件自身的
name
选项,如果name
选项不可用,则匹配它 的局部注册名称 (父组件components
选项的键值)。匿名组件不能被匹配。max:最多可以缓存多少组件实例。一旦这个数字达到了,在新实例被创建之前, 已缓存组件中最久没有被访问的实例会被销毁掉
注意:
<keep-alive>
不会在函数式组件中正常工作,因为它们没有缓存实例。
<!-- 基本 -->
<keep-alive>
<component :is="view"></component>
</keep-alive>
<!-- 多个条件判断的子组件 -->
<keep-alive>
<comp-a v-if="a > 1"></comp-a>
<comp-b v-else></comp-b>
</keep-alive>
<!-- 和 `<transition>` 一起使用 -->
<transition>
<keep-alive>
<component :is="view"></component>
</keep-alive>
</transition>
<!-- 逗号分隔字符串 -->
<keep-alive include="a,b">
<component :is="view"></component>
</keep-alive>
<!-- 正则表达式 (使用 `v-bind`) -->
<keep-alive :include="/a|b/">
<component :is="view"></component>
</keep-alive>
<!-- 数组 (使用 `v-bind`) -->
<keep-alive :include="['a', 'b']">
<component :is="view"></component>
</keep-alive>
<keep-alive>
是Vue
中内置的一个抽象组件,它自身不会渲染一个DOM
元素,也不会出现在父组件链中。当它包裹动态组件时,会缓存不活动的组件实例,而不是销毁它们。
keep-alive
是Vue的一个内置组件,也就是说在vue内部自己封装了一个组件;它会把需要缓存的组件保存起来,下次使用的时候直接从缓存中获取,而不是销毁和重新创建,因此可以保留组件的状态,从而可以提高性能;
源码预览
// 源码位置 src/core/components/keep-alive.js
function getComponentName (opts: ?VNodeComponentOptions): ?string {
return opts && (opts.Ctor.options.name || opts.tag)
}
function matches (pattern: string | RegExp | Array<string>, name: string): boolean {
if (Array.isArray(pattern)) {
return pattern.indexOf(name) > -1
} else if (typeof pattern === 'string') {
return pattern.split(',').indexOf(name) > -1
} else if (isRegExp(pattern)) {
return pattern.test(name)
}
/* istanbul ignore next */
return false
}
function pruneCache (keepAliveInstance: any, filter: Function) {
const { cache, keys, _vnode } = keepAliveInstance
for (const key in cache) {
const entry: ?CacheEntry = cache[key]
if (entry) {
const name: ?string = entry.name
if (name && !filter(name)) {
pruneCacheEntry(cache, key, keys, _vnode)
}
}
}
}
function pruneCacheEntry (
cache: CacheEntryMap,
key: string,
keys: Array<string>,
current?: VNode
) {
const entry: ?CacheEntry = cache[key]
if (entry && (!current || entry.tag !== current.tag)) {
entry.componentInstance.$destroy()
}
cache[key] = null
remove(keys, key)
}
const patternTypes: Array<Function> = [String, RegExp, Array]
export default {
name: 'keep-alive',
abstract: true,
props: {
include: patternTypes,
exclude: patternTypes,
max: [String, Number]
},
methods: {
cacheVNode() {
const { cache, keys, vnodeToCache, keyToCache } = this
if (vnodeToCache) {
const { tag, componentInstance, componentOptions } = vnodeToCache
cache[keyToCache] = {
name: getComponentName(componentOptions),
tag,
componentInstance,
}
keys.push(keyToCache)
// prune oldest entry
if (this.max && keys.length > parseInt(this.max)) {
pruneCacheEntry(cache, keys[0], keys, this._vnode)
}
this.vnodeToCache = null
}
}
},
created () {
this.cache = Object.create(null)
this.keys = []
},
destroyed () {
for (const key in this.cache) {
pruneCacheEntry(this.cache, key, this.keys)
}
},
mounted () {
this.cacheVNode()
this.$watch('include', val => {
pruneCache(this, name => matches(val, name))
})
this.$watch('exclude', val => {
pruneCache(this, name => !matches(val, name))
})
},
updated () {
this.cacheVNode()
},
render () {
const slot = this.$slots.default
const vnode: VNode = getFirstComponentChild(slot)
const componentOptions: ?VNodeComponentOptions = vnode && vnode.componentOptions
if (componentOptions) {
// check pattern
const name: ?string = getComponentName(componentOptions)
const { include, exclude } = this
if (
// not included
(include && (!name || !matches(include, name))) ||
// excluded
(exclude && name && matches(exclude, name))
) {
return vnode
}
const { cache, keys } = this
const key: ?string = vnode.key == null
// same constructor may get registered as different local components
// so cid alone is not enough (#3269)
? componentOptions.Ctor.cid + (componentOptions.tag ? `::${componentOptions.tag}` : '')
: vnode.key
if (cache[key]) {
vnode.componentInstance = cache[key].componentInstance
// make current key freshest
remove(keys, key)
keys.push(key)
} else {
// delay setting the cache until update
this.vnodeToCache = vnode
this.keyToCache = key
}
vnode.data.keepAlive = true
}
return vnode || (slot && slot[0])
}
}
首先定义props属性和对应的类型,include匹配需要缓存的组件名,exclude匹配不需要缓存的组件名,max匹配最多缓存的组件数量,如果超过这个数量就删除调最久不使用的组件;
props: {
include: patternTypes,
exclude: patternTypes,
max: [String, Number]
},
created生命周期钩子中,定义了cache用来保存缓存的组件,keys用来存储需要保存组件的key;
created () {
// 创建一个空的对象用来缓存组件
this.cache = Object.create(null)
// 缓存的组件的key
this.keys = []
},
此组件不是模板组件,而是一个函数组件,通过render函数返回html;render函数中首先获取了默认插槽(在keep-alive
标签之间放的值会被作为默认插入的值),获取插槽中的第一个节点(keep-alive
只缓存它内部的第一个组件),接着获取了节点上的options选项;
// 获取到默认插槽
const slot = this.$slots.default
// 获取到插槽中的第一个子节点
const vnode: VNode = getFirstComponentChild(slot)
// 获取到第一个子节点的options选项
const componentOptions: ?VNodeComponentOptions = vnode && vnode.componentOptions
接着判断options选项是否存在,存在就获取组件的名称,然后判断include存在并且name不存在或者name不在include中表示不需要缓存,或exclude存在并且name存在并且name存在exclude中表示不需要缓存,直接返回当前节点;
// componentOptions存在
if (componentOptions) {
// 获取到组件的名称
const name: ?string = getComponentName(componentOptions)
const { include, exclude } = this
if (
// include存在,并且没有name或name不存在include中 或 有exclude并且有name并且
// name存在于exclude中直接返回当前vnode
(include && (!name || !matches(include, name))) ||
// excluded
(exclude && name && matches(exclude, name))
) {
return vnode
}
接着获取到节点的key,通过key判断是否在缓存中,如果在缓存中,就获取到缓存中此节点的实例进行重新赋值;并且删除掉keys中的此节点的key,再重新添加到keys的尾部(目的:区分不常用的组件,不常用的组件慢慢的会移动到keys数组的头部)
const { cache, keys } = this
// 获取到节点的key
const key: ?string = vnode.key == null
// same constructor may get registered as different local components
// so cid alone is not enough (#3269)
? componentOptions.Ctor.cid + (componentOptions.tag ? `::${componentOptions.tag}` : '')
: vnode.key
// 如果存在缓存中
if (cache[key]) {
// 获取缓存中的值的componentInstance给当前节点的componentInstance
vnode.componentInstance = cache[key].componentInstance
// 从keys数组中删除掉当前的key
remove(keys, key)
// 重新添加到尾部
keys.push(key)
}
如果不存在于缓存中,就保存当前节点和它的key;在mounted生命钩子中进行缓存;
else { // 如果缓存中不存在
// delay setting the cache until update
// 赋值给vnodeToCache和keyToCache
this.vnodeToCache = vnode
this.keyToCache = key
}
最后给当前节点添加标记并且返回此节点
// 添加keepAlive属性为true
vnode.data.keepAlive = true
}
// 返回当前节点或插槽中的第一个节点
return vnode || (slot && slot[0])
render执行之后就执行mounted生命钩子,钩子中首先执行了cacheVNode方法,此方法主要用来缓存节点;方法中首先判断有没有要缓存的节点,如果有就添加到cache和keys中,接着判断是否有max,如果有并且当前的keys中缓存的值已经超过了设置的max的值,那么就把keys的第一项从缓存中进行删除(LRU
(Least recently used,最近最少使用)的缓存策略:核心就是最近被访问的数据后面访问的几率比较高,因此把最近访问的数据不断的从历史记录中取出来重新添加到尾部,久而久之开头位置的数据就是不常访问的因此优先进行删除);
mounted () {
// 缓存节点
this.cacheVNode()
...
},
methods: {
// 缓存节点
cacheVNode() {
const { cache, keys, vnodeToCache, keyToCache } = this
// 如果存在没有被缓存的节点
if (vnodeToCache) {
const { tag, componentInstance, componentOptions } = vnodeToCache
// 进行缓存
cache[keyToCache] = {
name: getComponentName(componentOptions),
tag,
componentInstance,
}
// 把key添加到keys中
keys.push(keyToCache)
// prune oldest entry
// 如果max存在,并且当前的keys的长度超过了max的值,那么就删除keys中的第一个节点
if (this.max && keys.length > parseInt(this.max)) {
pruneCacheEntry(cache, keys[0], keys, this._vnode)
}
// 设置为空
this.vnodeToCache = null
}
}
},
mounted中还监听了include和exclude两个属性,当值变化的时候就根据变化的值判断当前节点的name是否存在于include中,不存在就从缓存中进行删除;如果存在于exclude中就从缓存中进行删除;
// 监控include 执行pruneCache判断include的新值中是否存在name不存在就进行删除节点
this.$watch('include', val => {
pruneCache(this, name => matches(val, name))
})
// 监控exclude 执行pruneCache判断exclude的新值中是否存在name存在就进行删除节点
this.$watch('exclude', val => {
pruneCache(this, name => !matches(val, name))
})
// 通过pattern的类型判断name是否在它的里面
function matches (pattern: string | RegExp | Array<string>, name: string): boolean {
// 数组类型
if (Array.isArray(pattern)) {
return pattern.indexOf(name) > -1
} else if (typeof pattern === 'string') { // 字符串
return pattern.split(',').indexOf(name) > -1
} else if (isRegExp(pattern)) { // 正则
return pattern.test(name)
}
/* istanbul ignore next */
return false
}
function pruneCache (keepAliveInstance: any, filter: Function) {
const { cache, keys, _vnode } = keepAliveInstance
// 遍历缓存
for (const key in cache) {
const entry: ?CacheEntry = cache[key]
// 缓存中有这个值
if (entry) {
// 获取到name
const name: ?string = entry.name
// name存在并且不存在include或exclude中就进行删除
if (name && !filter(name)) {
pruneCacheEntry(cache, key, keys, _vnode)
}
}
}
}
destroyed函数,当keep-alive
被销毁的之后,遍历cache删除cache中的所有缓存
destroyed () {
// 遍历环缓存 进行删除和销毁
for (const key in this.cache) {
pruneCacheEntry(this.cache, key, this.keys)
}
},
update函数,当前keep-alive
更新的时候进行添加缓存
updated () {
this.cacheVNode()
},
总结:
- keep-alive是vue内部实现的一个函数组件,它自身不会被渲染成dom元素
- 在函数组件中有两个属性分别用来存储要缓存的组件的实例和它的key
- 在函数组件的render函数中,首先获取到默认插槽中的第一个元素作为渲染的元素
- 通过获取到includes和excludes属性,来判断当前组件的name是否在其中,从而判断是否需要被缓存
- 如果不需要缓存直接返回当前的虚拟dom
- 否则直接从缓存中找当前组件并且返回,并且把key从缓存中进行删除,再push到尾部(最近最久未被访问的组件就会慢慢的移动到缓存数组的头部);如果需要缓存就把当前的组件的实例采用LRU(Least Recently used)最近最小使用缓存原则;
- 当组件再次被渲染的时候直接使用缓存中的实例,无需重新进行创建组件的实例
转载自:https://juejin.cn/post/7239226209371652151