资讯专栏INFORMATION COLUMN

彻底揭秘keep-alive原理

lavnFan / 3512人阅读

摘要:我们留意到,这里不是简单地将置为,而是遍历调用函数删除。执行组件的钩子函数删除缓存还要对应执行组件实例的钩子函数。这个在不可忽视钩子函数章节会再次出场。参考技术揭秘源码

一、前言

</>复制代码

  1. 原文链接:github.com/qi...

本文介绍的内容包括:

keep-alive用法:动态组件&vue-router

keep-alive源码解析

keep-alive组件及其包裹组件的钩子

keep-alive组件及其包裹组件的渲染

二、keep-alive介绍与应用 2.1 keep-alive是什么

</>复制代码

  1. keep-alive是一个抽象组件:它自身不会渲染一个 DOM 元素,也不会出现在父组件链中;使用keep-alive包裹动态组件时,会缓存不活动的组件实例,而不是销毁它们。

2.2 一个场景

用户在某个列表页面选择筛选条件过滤出一份数据列表,由列表页面进入数据详情页面,再返回该列表页面,我们希望:列表页面可以保留用户的筛选(或选中)状态。keep-alive就是用来解决这种场景。当然keep-alive不仅仅是能够保存页面/组件的状态这么简单,它还可以避免组件反复创建和渲染,有效提升系统性能。 总的来说,keep-alive用于保存组件的渲染状态。

2.3 keep-alive用法

在动态组件中的应用

</>复制代码

  1. <keep-alive :include="whiteList" :exclude="blackList" :max="amount">
  2. <component :is="currentComponent">component>
  3. keep-alive>

在vue-router中的应用

</>复制代码

  1. <keep-alive :include="whiteList" :exclude="blackList" :max="amount">
  2. <router-view>router-view>
  3. keep-alive>

include定义缓存白名单,keep-alive会缓存命中的组件;exclude定义缓存黑名单,被命中的组件将不会被缓存;max定义缓存组件上限,超出上限使用LRU的策略置换缓存数据。

三、源码剖析

keep-alive.js内部另外还定义了一些工具函数,我们按住不表,先看它对外暴露的对象。

</>复制代码

  1. // src/core/components/keep-alive.js
  2. export default {
  3. name: "keep-alive",
  4. abstract: true, // 判断当前组件虚拟dom是否渲染成真是dom的关键
  5. props: {
  6. include: patternTypes, // 缓存白名单
  7. exclude: patternTypes, // 缓存黑名单
  8. max: [String, Number] // 缓存的组件实例数量上限
  9. },
  10. created () {
  11. this.cache = Object.create(null) // 缓存虚拟dom
  12. this.keys = [] // 缓存的虚拟dom的健集合
  13. },
  14. destroyed () {
  15. for (const key in this.cache) { // 删除所有的缓存
  16. pruneCacheEntry(this.cache, key, this.keys)
  17. }
  18. },
  19. mounted () {
  20. // 实时监听黑白名单的变动
  21. this.$watch("include", val => {
  22. pruneCache(this, name => matches(val, name))
  23. })
  24. this.$watch("exclude", val => {
  25. pruneCache(this, name => !matches(val, name))
  26. })
  27. },
  28. render () {
  29. // 先省略...
  30. }
  31. }

可以看出,与我们定义组件的过程一样,先是设置组件名为keep-alive,其次定义了一个abstract属性,值为true。这个属性在vue的官方教程并未提及,却至关重要,后面的渲染过程会用到。props属性定义了keep-alive组件支持的全部参数。

keep-alive在它生命周期内定义了三个钩子函数:

created

初始化两个对象分别缓存VNode(虚拟DOM)和VNode对应的键集合

destroyed

删除this.cache中缓存的VNode实例。我们留意到,这里不是简单地将this.cache置为null,而是遍历调用pruneCacheEntry函数删除。

</>复制代码

  1. // src/core/components/keep-alive.js
  2. function pruneCacheEntry (
  3. cache: VNodeCache,
  4. key: string,
  5. keys: Array,
  6. current");) {
  7. const cached = cache[key]
  8. if (cached && (!current || cached.tag !== current.tag)) {
  9. cached.componentInstance.$destroy() // 执行组件的destory钩子函数
  10. }
  11. cache[key] = null
  12. remove(keys, key)
  13. }

删除缓存VNode还要对应执行组件实例的destory钩子函数。

mounted

mounted这个钩子中对includeexclude参数进行监听,然后实时地更新(删除)this.cache对象数据。pruneCache函数的核心也是去调用pruneCacheEntry

render

</>复制代码

  1. // src/core/components/keep-alive.js
  2. render () {
  3. const slot = this.$slots.default
  4. const vnode: VNode = getFirstComponentChild(slot) // 找到第一个子组件对象
  5. const componentOptions: ");if (componentOptions) { // 存在组件参数
  6. // check pattern
  7. const name: ");// 组件名
  8. const { include, exclude } = this
  9. if ( // 条件匹配
  10. // not included
  11. (include && (!name || !matches(include, name))) ||
  12. // excluded
  13. (exclude && name && matches(exclude, name))
  14. ) {
  15. return vnode
  16. }
  17. const { cache, keys } = this
  18. const key: ");null // 定义组件的缓存key
  19. // same constructor may get registered as different local components
  20. // so cid alone is not enough (#3269)
  21. ");`::${componentOptions.tag}` : "")
  22. : vnode.key
  23. if (cache[key]) { // 已经缓存过该组件
  24. vnode.componentInstance = cache[key].componentInstance
  25. // make current key freshest
  26. remove(keys, key)
  27. keys.push(key) // 调整key排序
  28. } else {
  29. cache[key] = vnode // 缓存组件对象
  30. keys.push(key)
  31. // prune oldest entry
  32. if (this.max && keys.length > parseInt(this.max)) { // 超过缓存数限制,将第一个删除
  33. pruneCacheEntry(cache, keys[0], keys, this._vnode)
  34. }
  35. }
  36. vnode.data.keepAlive = true // 渲染和执行被包裹组件的钩子函数需要用到
  37. }
  38. return vnode || (slot && slot[0])
  39. }

第一步:获取keep-alive包裹着的第一个子组件对象及其组件名;

第二步:根据设定的黑白名单(如果有)进行条件匹配,决定是否缓存。不匹配,直接返回组件实例(VNode),否则执行第三步;

第三步:根据组件ID和tag生成缓存Key,并在缓存对象中查找是否已缓存过该组件实例。如果存在,直接取出缓存值并更新该keythis.keys中的位置(更新key的位置是实现LRU置换策略的关键),否则执行第四步;

第四步:在this.cache对象中存储该组件实例并保存key值,之后检查缓存的实例数量是否超过max的设置值,超过则根据LRU置换策略删除最近最久未使用的实例(即是下标为0的那个key)。

第五步:最后并且很重要,将该组件实例的keepAlive属性值设置为true。这个在@不可忽视:钩子函数 章节会再次出场。

四、重头戏:渲染 4.1 Vue的渲染过程

借一张图看下Vue渲染的整个过程:

Vue的渲染是从图中的render阶段开始的,但keep-alive的渲染是在patch阶段,这是构建组件树(虚拟DOM树),并将VNode转换成真正DOM节点的过程。

简单描述从renderpatch的过程

我们从最简单的new Vue开始:

</>复制代码

  1. import App from "./App.vue"
  2. new Vue({
  3. render: h => h(App),
  4. }).$mount("#app")

Vue在渲染的时候先调用原型上的_render函数将组件对象转化为一个VNode实例;而_render是通过调用createElementcreateEmptyVNode两个函数进行转化;

createElement的转化过程会根据不同的情形选择new VNode或者调用createComponent函数做VNode实例化;

完成VNode实例化后,这时候Vue调用原型上的_update函数把VNode渲染为真实DOM,这个过程又是通过调用__patch__函数完成的(这就是pacth阶段了)

用一张图表达:

4.2 keep-alive组件的渲染

我们用过keep-alive都知道,它不会生成真正的DOM节点,这是怎么做到的?

</>复制代码

  1. // src/core/instance/lifecycle.js
  2. export function initLifecycle (vm: Component) {
  3. const options = vm.$options
  4. // 找到第一个非abstract的父组件实例
  5. let parent = options.parent
  6. if (parent && !options.abstract) {
  7. while (parent.$options.abstract && parent.$parent) {
  8. parent = parent.$parent
  9. }
  10. parent.$children.push(vm)
  11. }
  12. vm.$parent = parent
  13. // ...
  14. }

Vue在初始化生命周期的时候,为组件实例建立父子关系会根据abstract属性决定是否忽略某个组件。在keep-alive中,设置了abstract: true,那Vue就会跳过该组件实例。

最后构建的组件树中就不会包含keep-alive组件,那么由组件树渲染成的DOM树自然也不会有keep-alive相关的节点了。

keep-alive包裹的组件是如何使用缓存的?

patch阶段,会执行createComponent函数:

</>复制代码

  1. // src/core/vdom/patch.js
  2. function createComponent (vnode, insertedVnodeQueue, parentElm, refElm) {
  3. let i = vnode.data
  4. if (isDef(i)) {
  5. const isReactivated = isDef(vnode.componentInstance) && i.keepAlive
  6. if (isDef(i = i.hook) && isDef(i = i.init)) {
  7. i(vnode, false /* hydrating */)
  8. }
  9. if (isDef(vnode.componentInstance)) {
  10. initComponent(vnode, insertedVnodeQueue)
  11. insert(parentElm, vnode.elm, refElm) // 将缓存的DOM(vnode.elm)插入父元素中
  12. if (isTrue(isReactivated)) {
  13. reactivateComponent(vnode, insertedVnodeQueue, parentElm, refElm)
  14. }
  15. return true
  16. }
  17. }
  18. }

在首次加载被包裹组件时,由keep-alive.js中的render函数可知,vnode.componentInstance的值是undefinedkeepAlive的值是true,因为keep-alive组件作为父组件,它的render函数会先于被包裹组件执行;那么就只执行到i(vnode, false /* hydrating */),后面的逻辑不再执行;

再次访问被包裹组件时,vnode.componentInstance的值就是已经缓存的组件实例,那么会执行insert(parentElm, vnode.elm, refElm)逻辑,这样就直接把上一次的DOM插入到了父元素中。

五、不可忽视:钩子函数 5.1 只执行一次的钩子

一般的组件,每一次加载都会有完整的生命周期,即生命周期里面对应的钩子函数都会被触发,为什么被keep-alive包裹的组件却不是呢? 我们在@源码剖析 章节分析到,被缓存的组件实例会为其设置keepAlive = true,而在初始化组件钩子函数中:

</>复制代码

  1. // src/core/vdom/create-component.js
  2. const componentVNodeHooks = {
  3. init (vnode: VNodeWithData, hydrating: boolean): ");if (
  4. vnode.componentInstance &&
  5. !vnode.componentInstance._isDestroyed &&
  6. vnode.data.keepAlive
  7. ) {
  8. // kept-alive components, treat as a patch
  9. const mountedNode: any = vnode // work around flow
  10. componentVNodeHooks.prepatch(mountedNode, mountedNode)
  11. } else {
  12. const child = vnode.componentInstance = createComponentInstanceForVnode(
  13. vnode,
  14. activeInstance
  15. )
  16. child.$mount(hydrating ");undefined, hydrating)
  17. }
  18. }
  19. // ...
  20. }

可以看出,当vnode.componentInstancekeepAlive同时为truly值时,不再进入$mount过程,那mounted之前的所有钩子函数(beforeCreatecreatedmounted)都不再执行。

5.2 可重复的activated

patch的阶段,最后会执行invokeInsertHook函数,而这个函数就是去调用组件实例(VNode)自身的insert钩子:

</>复制代码

  1. // src/core/vdom/patch.js
  2. function invokeInsertHook (vnode, queue, initial) {
  3. if (isTrue(initial) && isDef(vnode.parent)) {
  4. vnode.parent.data.pendingInsert = queue
  5. } else {
  6. for (let i = 0; i < queue.length; ++i) {
  7. queue[i].data.hook.insert(queue[i]) // 调用VNode自身的insert钩子函数
  8. }
  9. }
  10. }

再看insert钩子:

</>复制代码

  1. // src/core/vdom/create-component.js
  2. const componentVNodeHooks = {
  3. // init()
  4. insert (vnode: MountedComponentVNode) {
  5. const { context, componentInstance } = vnode
  6. if (!componentInstance._isMounted) {
  7. componentInstance._isMounted = true
  8. callHook(componentInstance, "mounted")
  9. }
  10. if (vnode.data.keepAlive) {
  11. if (context._isMounted) {
  12. queueActivatedComponent(componentInstance)
  13. } else {
  14. activateChildComponent(componentInstance, true /* direct */)
  15. }
  16. }
  17. // ...
  18. }

在这个钩子里面,调用了activateChildComponent函数递归地去执行所有子组件的activated钩子函数:

</>复制代码

  1. // src/core/instance/lifecycle.js
  2. export function activateChildComponent (vm: Component, direct");) {
  3. if (direct) {
  4. vm._directInactive = false
  5. if (isInInactiveTree(vm)) {
  6. return
  7. }
  8. } else if (vm._directInactive) {
  9. return
  10. }
  11. if (vm._inactive || vm._inactive === null) {
  12. vm._inactive = false
  13. for (let i = 0; i < vm.$children.length; i++) {
  14. activateChildComponent(vm.$children[i])
  15. }
  16. callHook(vm, "activated")
  17. }
  18. }

相反地,deactivated钩子函数也是一样的原理,在组件实例(VNode)的destroy钩子函数中调用deactivateChildComponent函数。

参考

Vue技术揭秘|keep-alive

Vue源码

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/7314.html

相关文章

  • 彻底隐藏Nginx版本号的安全性与方法

    摘要:默认是显示版本号的,如这样就给人家看到你的服务器版本是,前些时间暴出了一些版本漏洞,就是说有些版本有漏洞,而有些版本没有。这样暴露出来的版本号就容易变成攻击者可利用的信息。 Nginx默认是显示版本号的,如: [root@hadooptest ~]# curl -I www.nginx.org HTTP/1.1 200 OK Server: nginx/0.8.44 Date: Tu...

    starsfun 评论0 收藏0
  • 彻底隐藏Nginx版本号的安全性与方法

    摘要:默认是显示版本号的,如这样就给人家看到你的服务器版本是,前些时间暴出了一些版本漏洞,就是说有些版本有漏洞,而有些版本没有。这样暴露出来的版本号就容易变成攻击者可利用的信息。 Nginx默认是显示版本号的,如: [root@hadooptest ~]# curl -I www.nginx.org HTTP/1.1 200 OK Server: nginx/0.8.44 Date: Tu...

    GHOST_349178 评论0 收藏0
  • 详解Vue.js 技术

    本文主要从8个章节详解vue技术揭秘,小编觉得挺有用的,分享给大家。 为了把 Vue.js 的源码讲明白,课程设计成由浅入深,分为核心、编译、扩展、生态四个方面去讲,并拆成了八个章节,如下: 准备工作 Introduction 认识 Flow Vue.js 源码目录设计 Vue.js 源码构建 从入口开始 数据驱动 Introduction new Vue 发生了什么 Vue ...

    saucxs 评论0 收藏0

发表评论

0条评论

lavnFan

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<