组件的 keep-alive 简介
本篇文章,我们来讲一下keep-alive
的实现。 更容易看懂
Vue
中,有三个内置的抽象组件,分别是keep-alive
、transition
和transition-group
,
它们都有一个共同的特点,就是自身不会渲染一个DOM元素,也不会出现在父组件链中。
keep-alive
的作用,是包裹动态组件时,会缓存不活动的组件实例,而不是销毁它们。具体的用法见这里。
该组件的定义,是在src/core/components/keep-alive.js
文件中。
它会在Vue
初始化时,添加在Vue.options.components
上,所以在所有的组件中,都可以直接只用它。
直接看代码:
export default {
name: 'keep-alive',
abstract: true, props: {
...
}, created () {
this.cache = Object.create(null)
}, destroyed () {
...
}, watch: {
...
}, render () {
...
}
}
name
不用多说,abstract: true
这个条件我们自己定义组件时通常不会用,
它是用来标识当前的组件是一个抽象组件,它自身不会渲染一个真实的DOM元素。
比如在创建两个vm
实例之间的父子关系时,会跳过抽象组件的实例:
let parent = options.parent
if (parent && !options.abstract) {
while (parent.$options.abstract && parent.$parent) {
parent = parent.$parent
}
parent.$children.push(vm)
}
props
表示我们可以传入include
来匹配哪些组件可以缓存,exclude
来匹配哪些组件不缓存。
created
钩子函数调用时,会创建一个this.cache
对象用于缓存它的子组件。
destroyed
表示keep-alive
被销毁时,会同时销毁它缓存的组件,并调用deactivated
钩子函数。
function pruneCacheEntry (vnode: ?VNode) {
if (vnode) {
if (!vnode.componentInstance._inactive) {
callHook(vnode.componentInstance, 'deactivated')
}
vnode.componentInstance.$destroy()
}
}
watch
是在我们改变props
传入的值时,同时对this.cache
缓存中的数据进行处理。
function pruneCache (cache: VNodeCache, filter: Function) {
for (const key in cache) {
const cachedNode: ?VNode = cache[key]
if (cachedNode) {
const name: ?string = getComponentName(cachedNode.componentOptions)
if (name && !filter(name)) {
pruneCacheEntry(cachedNode)
cache[key] = null
}
}
}
}
抽象组件没有实际的DOM元素,所以也就没有template
模板,它会有一个render
函数,我们就来看看里面进行了哪些操作。
render () {
const vnode: VNode = getFirstComponentChild(this.$slots.default)
const componentOptions: ?VNodeComponentOptions = vnode && vnode.componentOptions
if (componentOptions) {
// check pattern
const name: ?string = getComponentName(componentOptions)
if (name && (
(this.include && !matches(this.include, name)) ||
(this.exclude && matches(this.exclude, name))
)) {
return vnode
}
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 (this.cache[key]) {
vnode.componentInstance = this.cache[key].componentInstance
} else {
this.cache[key] = vnode
}
vnode.data.keepAlive = true
}
return vnode
}
首先,调用getFirstComponentChild
方法,来获取this.$slots.default
中的第一个元素。
export function getFirstComponentChild (children: ?Array<VNode>): ?VNode {
return children && children.filter((c: VNode) => c && c.componentOptions)[0]
}
this.$slots.default
中包含的是什么内容,我们在《slot和作用域插槽》中已经详细的做了讲解。
从上面的方法我们可以看到,在我们会过滤掉非自定义的标签,然后获取第一个自定义标签所对应的vnode
。
所以,如果keep-alive
里面包裹的是html
标签,是不会渲染的。
然后获取componentOptions
,
vdom——VNode中我们介绍过componentOptions
包含五个元素
{ Ctor, propsData, listeners, tag, children }
。
function getComponentName (opts: ?VNodeComponentOptions): ?string {
return opts && (opts.Ctor.options.name || opts.tag)
}
通过getComponentName
方法来获取组件名,然后判断该组件是否合法,
如果include
不匹配或exclude
匹配,则说明该组件不需要缓存,
此时直接返回该vnode
。
否则,vnode.key
不存在则生成一个,存在则就用vnode.key
作为key
。
然后把该vnode
添加到this.cache
中,并设置vnode.data.keepAlive = true
。
最终返回该vnode
。
以上只是render
函数执行的过程,keep-alive
本身也是一个组件,
在render
函数调用生成vnode
后,同样会走__patch__
。在创建和diff
的过程中,
也会调用init
、prepatch
、insert
和destroy
钩子函数。
不过,每个钩子函数中所做的处理,和普通组件有所不同。
init (
vnode: VNodeWithData,
hydrating: boolean,
parentElm: ?Node,
refElm: ?Node
): ?boolean {
if (!vnode.componentInstance || vnode.componentInstance._isDestroyed) {
const child = vnode.componentInstance = createComponentInstanceForVnode(
vnode,
activeInstance,
parentElm,
refElm
)
child.$mount(hydrating ? vnode.elm : undefined, hydrating)
} else if (vnode.data.keepAlive) {
// kept-alive components, treat as a patch
const mountedNode: any = vnode // work around flow
componentVNodeHooks.prepatch(mountedNode, mountedNode)
}
},
在keep-alive
组件内调用__patch__
时,如果render
返回的vnode
是第一次使用,
则走正常的创建流程,如果之前创建过且添加了vnode.data.keepAlive
,
则直接调用prepatch
方法,且传入的新旧vnode
相同。
prepatch (oldVnode: MountedComponentVNode, vnode: MountedComponentVNode) {
const options = vnode.componentOptions
const child = vnode.componentInstance = oldVnode.componentInstance
updateChildComponent(
child,
options.propsData, // updated props
options.listeners, // updated listeners
vnode, // new parent vnode
options.children // new children
)
},
prepatch
函数做了哪些工作,之前也详细的介绍过,这里就不多说了。
简单的总结,就是依据新vnode
中的数据,更新组件内容。
insert (vnode: MountedComponentVNode) {
if (!vnode.componentInstance._isMounted) {
vnode.componentInstance._isMounted = true
callHook(vnode.componentInstance, 'mounted')
}
if (vnode.data.keepAlive) {
activateChildComponent(vnode.componentInstance, true /* direct */)
}
},
在组件插入到页面后,如果是vnode.data.keepAlive
则会调用activateChildComponent
,
这里面主要是调用子组件的activated
钩子函数,并设置vm._inactive
的标识状态。
destroy (vnode: MountedComponentVNode) {
if (!vnode.componentInstance._isDestroyed) {
if (!vnode.data.keepAlive) {
vnode.componentInstance.$destroy()
} else {
deactivateChildComponent(vnode.componentInstance, true /* direct */)
}
}
}
在组件销毁时,如果是vnode.data.keepAlive
返回true
,
则只调用deactivateChildComponent
,这里面主要是调用子组件的deactivated
钩子函数,
并设置vm._directInactive
的标识状态。因为vnode.data.keepAlive
为true
的组件,
是会被keep-alive
缓存起来的,所以不会直接调用它的$destroy()
方法,
上面我们也提到了,当keep-alive
组件被销毁时,会触发它缓存中所有组件的$destroy()
。
因为keep-alive
包裹的组件状态变化,还会触发其子组件的activated
或deactivated
钩子函数,
activateChildComponent
和deactivateChildComponent
也会做一些这方面的处理,细节大家可以自行查看。
组件的 keep-alive 简介的更多相关文章
- 构建简单的 C++ 服务组件,第 1 部分: 服务组件体系结构 C++ API 简介
构建简单的 C++ 服务组件,第 1 部分: 服务组件体系结构 C++ API 简介 熟悉将用于 Apache Tuscany SCA for C++ 的 API.您将通过本文了解该 API 的主要组 ...
- React组件和生命周期简介
React 简介----React 是 Facebook 出品的一套颠覆式的前端开发类库.为什么说它是颠覆式的呢? 内存维护虚拟 DOM 对于传统的 DOM 维护,我们的步骤可能是:1.初始化 ...
- iOS组件化开发-CocoaPods简介
CocoaPods简介 任何一门开发语言到达一定阶段就会出现第三方的类库管理工具,比如Java的Maven.WEB的Webpack等.在iOS中类库的管理工具-CocoaPods. 利用CocoaPo ...
- Akka(17): Stream:数据流基础组件-Source,Flow,Sink简介
在大数据程序流行的今天,许多程序都面临着共同的难题:程序输入数据趋于无限大,抵达时间又不确定.一般的解决方法是采用回调函数(callback-function)来实现的,但这样的解决方案很容易造成“回 ...
- .Net服务组件(ServicedComponent)简介及其使用
.NET Enterprise Services 为企业应用程序提供重要的基础结构.COM+ 为企业环境中部署的组件编程模型提供服务结构.System.EnterpriseServices命名空间向 ...
- .NET Core 基于Quartz的UI可视化操作组件 GZY.Quartz.MUI 简介
前言 最近在用Quartz做定时任务.虽然很方便,但是Quartz自己貌似是没有UI界面的..感觉操作起来 就很难受.. 查了一下,貌似有个UI组件 不过看了一下文档..直接给我劝退了..太麻烦了 我 ...
- 2.第一篇 k8s组件版本及功能简介
文章转载自:https://mp.weixin.qq.com/s?__biz=MzI1MDgwNzQ1MQ==&mid=2247483772&idx=1&sn=a693d8a9 ...
- (day20)javaEE三大组件之一Servlet (简介(二)servletconfig,servletContext,session,cookie,request,response,out)
javaEE是服务器编程,javaEE提供了服务器的接口让具体的服务器去创建实现的对象 JavaEE是sun公司为了解决企业级开发定义的一套技术,只提供了规范,具体的实现是由服务器完成的 servle ...
- day18(javaEE三大组件之一servlet(简介(一)))
Servlet servlet是小型服务器语言,使用它可以处理前台传递来的信息,servlet进行处理后在响应给前台,其中servlet起到了关键性的作用.前端输入的信息可以持久化的存储在数据库中,并 ...
- VMware vSphere 组件和功能
https://pubs.vmware.com/vsphere-50/index.jsp?topic=%2Fcom.vmware.vsphere.introduction.doc_50%2FGUID- ...
随机推荐
- [Shiro] - Shiro之进阶
上一个知识点ssm, 哪里需要权限,哪里写注解@RequirePermission就行. 但是,真正项目开发的时候, 这种方式就很有局限性了, 当权限配置关系发生变化, 每次都要修改代码, 编译打包重 ...
- [Linux] - Linux安装JDK
https://www.oracle.com/technetwork/java/javase/downloads/jdk8-downloads-2133151.html <官方JDK下载 之后 ...
- [POJ1958][Strange Tower of Hanoi]
题目描述 求解 \(n\) 个盘子 \(4\) 座塔的 Hanoi 问题最少需要多少步 问题分析 考虑 \(3\) 座塔的 Hanoi 问题,记 \(f[i]\) 表示最少需要多少步, 则 \(f[i ...
- MVC ---- DBHelper.ttinclude
在通过T4模版引擎之基础入门 对T4有了初步印象后,我们开始实战篇.T4模板引擎可以当做一个代码生成器,代码生成器的职责当然是用来生成代码(这不是废话吗).而这其中我们使用的最普遍的是根据数据库生成实 ...
- Linux 普通用户拿到root权限及使用szrz命令上传下载文件
1.如何拿到root权限 在shell里如果看到你的命令输入行最前面显示的是 $符号,说明目前账号只有系统的普通权限. 输入:sudo su 这时能看到shell的输入行最前面已经显示的是#号,说明已 ...
- 用java代码将数组元素顺序颠倒
package test; public class Recover { public int[] reverse(int[] a) { int[] b = new int[a.length]; in ...
- 《A_Pancers》团队项目用户验收评审
团队项目用户验收评审 一.关于源代码管理的10 个问题: 1.你的团队的源代码控制在哪里?用的是什么系统?如何处理文件的锁定问题? 我们的项目都在github上面,用的win10系统,并且我们的文件没 ...
- 《Blue_Flke》团队项目软件系统设计改进
团队项目系统设计改进: 1.分析项目系统设计说明书初稿的不足,特别是软件系统结构模型建模不完善内容 在上一次的项目系统设计说明书中没有很好的完成软件系统结构模型的建模设计,只做了基本的系统项目原型模型 ...
- React生命周期钩子
最近的工作都很忙,所以很少完整的时间可以用来总结和回顾知识点,今天就趁着是周末,我准备在这里复习和回顾一下React的基础.工作中主要用的vue比较多,在工作中使用React也已经是一年前了,当时用的 ...
- Redis之有序集合命令
Redis 有序集合(sorted set) Redis 有序集合和集合一样也是string类型元素的集合,且不允许重复的成员. 不同的是每个元素都会关联一个double类型的分数.redis正是通过 ...