Vue3 KeepAlive实现原理解析

 更新时间:2022年09月05日 17:04:02   作者:Axin  
KeepAlive 是一个内置组件,那封装一个组件对于大家来说应该不会有太大的困难,它的核心逻辑在于它的 render 函数,它用 map 去记录要缓存的组件,就是 [key,vnode] 的形式,这篇文章主要介绍了Vue3 KeepAlive实现原理,需要的朋友可以参考下

思路

首先我们知道 KeepAlive 是一个内置组件,那封装一个组件对于大家来说应该不会有太大的困难,它的核心逻辑在于它的 render 函数,它用 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) { // 节点存在缓存中
    // copy over mounted state(复制过去的状态)
    vnode.el = cachedVNode.el
    vnode.component = cachedVNode.component

    // 将节点放到队尾
    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调用

tips:这两块逻辑是在 diff 中,看不懂没关系,只要注意一个点就是被缓存的组件节点有一个标识 ShapeFlags.COMPONENT_SHOULD_KEEP_ALIVE,我们在挂载组件的时候和卸载组件的时候判断一下是否有这个标识,有就去调用 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实现原理的文章就介绍到这了,更多相关Vue3 KeepAlive原理内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 使用idea创建第一个Vue项目

    使用idea创建第一个Vue项目

    最近在学习vue,本文主要介绍了使用idea创建第一个Vue项目,文中根据图文介绍的十分详尽,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-03-03
  • 图文详解vue框架安装步骤

    图文详解vue框架安装步骤

    我们在本篇内容里给大家整理了关于vue框架安装的步骤以及需要注意的地方,有需要的朋友们学习下。
    2019-02-02
  • Vue3处理模板和渲染函数的示例代码

    Vue3处理模板和渲染函数的示例代码

    Vue.js是一个流行的前端框架,以其易于学习和使用而闻名,在Vue3中,借助于Composition API和新的setup语法糖,模板和渲染函数的使用变得更加灵活和强大,在这篇博客中,我们将深入探讨Vue3是如何处理模板和渲染函数的,并通过示例代码来展示如何有效利用这些功能
    2024-11-11
  • 使用Vue调取接口,并渲染数据的示例代码

    使用Vue调取接口,并渲染数据的示例代码

    今天小编就为大家分享一篇使用Vue调取接口,并渲染数据的示例代码,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2019-10-10
  • Vue.js使用axios动态获取response里的data数据操作

    Vue.js使用axios动态获取response里的data数据操作

    这篇文章主要介绍了Vue.js使用axios动态获取response里的data数据操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-09-09
  • 使用axios发送post请求,将JSON数据改为form类型的示例

    使用axios发送post请求,将JSON数据改为form类型的示例

    今天小编就为大家分享一篇使用axios发送post请求,将JSON数据改为form类型的示例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2019-10-10
  • vue中post请求以a=a&b=b 的格式写遇到的问题

    vue中post请求以a=a&b=b 的格式写遇到的问题

    这篇文章主要介绍了vue中post请求以a=a&b=b 的格式写遇到的问题,需要的朋友可以参考下
    2018-04-04
  • vue post application/x-www-form-urlencoded如何实现传参

    vue post application/x-www-form-urlencoded如何实现传参

    这篇文章主要介绍了vue post application/x-www-form-urlencoded如何实现传参问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-04-04
  • vuex state及mapState的基础用法详解

    vuex state及mapState的基础用法详解

    这篇文章主要介绍了vuex state及mapState的基础用法详解,本文通过实例代码相结合的形式给大家介绍的非常详细,需要的朋友跟随脚本之家小编一起学习吧
    2018-04-04
  • Vue2升级到Vue3的教程及避坑指南

    Vue2升级到Vue3的教程及避坑指南

    随着Vue3的日益成熟,越来越多的团队开始考虑将现有Vue2项目升级到Vue3,作为一次重大版本更新,Vue3在带来性能提升和新特性的同时,也引入了一些不兼容的变化,本文将全面解析Vue2到Vue3升级过程中需要注意的关键点,帮助你顺利完成迁移,需要的朋友可以参考下
    2025-04-04

最新评论