组件的渲染过程中创建了一个带副作用的渲染函数,当数据变化的时候就会执行这个渲染函数来触发组件的更新。
副作用渲染函数更新组件的过程
更新组件主要做三件事情:更新组件 vnode 节点(initialTree)、渲染新的子树 vnode(subTree)、根据新旧子树 vnode 执行 patch 逻辑。
patch流程:
首先判断新旧节点是否是相同的 vnode 类型,如果不同,比如一个 div 更新成一个 ul,那么最简单的操作就是删除旧的 div 节点,再去挂载新的 ul 节点。
如果是相同的 vnode 类型,就需要走 diff 更新流程了,接着会根据不同的 vnode 类型执行不同的处理逻辑,这里我们仍然只分析普通元素类型和组件类型的处理过程。
加入App组件中:
|
|
如果msg发生了变化,那么首先触发app组件的更新,app组件的子树subtree是一个普通div,所以触发processElement逻辑,然后因为深度遍历,遇到hello这个节点的时候,会执行processComponent逻辑:
|
|
processComponent 主要通过执行 updateComponent 函数来更新子组件,updateComponent 函数在更新子组件的时候,会先执行 shouldUpdateComponent 函数,根据新旧子组件 vnode 来判断是否需要更新子组件。这里你只需要知道,在 shouldUpdateComponent 函数的内部,主要是通过检测和对比组件 vnode 中的 props、chidren、dirs、transiton 等属性,来决定子组件是否需要更新。
如果shouldUpdateComponent 返回true,那么会先执行invalidateJob避免一些不必要的更新,然后执行instance.update来主动触发子组件的更新。
回顾副作用函数(setupRenderEffect)的一部分代码:
|
|
结合上面的代码,我们在更新组件的 DOM 前,需要先更新组件 vnode 节点信息,包括更改组件实例的 vnode 指针、更新 props 和更新插槽等一系列操作,因为组件在稍后执行 renderComponentRoot 时会重新渲染新的子树 vnode ,它依赖了更新后的组件 vnode 中的 props 和 slots 等数据。
现在我们知道了一个组件重新渲染可能会有两种场景,一种是组件本身的数据变化,这种情况下 next 是 null;另一种是父组件在更新的过程中,遇到子组件节点,先判断子组件是否需要更新,如果需要则主动执行子组件的重新渲染方法,这种情况下 next 就是新的子组件 vnode。
所以processComponent本质上是去判断vnode中子组件是否需要更新,如果不需要,那就仅仅是更新一些vnode的属性,并让子组件保留对先的父组件vnode的引用,方便下次子组件更新的时候可以拿到父组件。
不过组件是抽象的,所以的组件即使经过深层递归后,最终也都会落到DOM元素的更新,所以接下来讨论普通元素的处理流程:
先看上文提到的processElement
|
|
更新DOM的流程比较简单,就是更新props和子节点,因为一个DOM元素一般也就是由这两部分构成的。
更新子节点是一个相对复杂的过程,一个子节点一般会有三种情况:纯文本、vnode数组和空。所以新旧节点排列组合就可能有9种情况。
而最为复杂的就是在都是数组的情况下,需要进行完整的diff算法。