React和DOM的那些事-节点删除算法

开发 前端 算法
本篇是详细解读React DOM操作的第一篇文章,文章所讲的内容发生在commit阶段。

[[378076]]

 本篇是详细解读React DOM操作的第一篇文章,文章所讲的内容发生在commit阶段。

Fiber架构使得React需要维护两类树结构,一类是Fiber树,另一类是DOM树。当删除DOM节点时,Fiber树也要同步变化。但请注意删除操作执行的时机:在完成DOM节点的其他变化(增、改)前,要先删除fiber节点,避免其他操作被干扰。 这是因为进行其他DOM操作时需要循环fiber树,此时如果有需要删除的fiber节点却还没删除的话,就会发生混乱。 

  1. function commitMutationEffects(  
  2.   firstChild: Fiber,  
  3.   root: FiberRoot,  
  4.   renderPriorityLevel,  
  5. ) {  
  6.   let fiber = firstChild 
  7.   while (fiber !== null) { 
  8.      // 首先进行删除  
  9.     const deletions = fiber.deletions;  
  10.     if (deletions !== null) {  
  11.       commitMutationEffectsDeletions(deletions, root, renderPriorityLevel);  
  12.     }  
  13.     // 如果删除之后的fiber还有子节点,  
  14.     // 递归调用commitMutationEffects来处理  
  15.     if (fiber.child !== null) {  
  16.       const primarySubtreeTag = fiber.subtreeTag & MutationSubtreeTag;  
  17.       if (primarySubtreeTag !== NoSubtreeTag) {  
  18.         commitMutationEffects(fiber.child, root, renderPriorityLevel);  
  19.       }  
  20.     }  
  21.     if (__DEV__) {/*...*/} else {  
  22.       // 执行其他DOM操作  
  23.       try {  
  24.         commitMutationEffectsImpl(fiber, root, renderPriorityLevel);  
  25.       } catch (error) {  
  26.         captureCommitPhaseError(fiber, error);  
  27.       }  
  28.     }  
  29.     fiberfiber = fiber.sibling;  
  30.   }  

fiber.deletions是render阶段的diff过程检测到fiber的子节点如果有需要被删除的,就会被加到这里来。

commitDeletion函数是删除节点的入口,它通过调用unmountHostComponents实现删除。搞懂删除操作之前,先看看场景。

有如下的Fiber树,Node(Node是一个代号,并不指的某个具体节点)节点即将被删除。   

  1. Fiber树  
  2.    div#root  
  3.       |  
  4.     <App/>  
  5.       |  
  6.      div  
  7.       |  
  8.    <Parent/>  
  9.       |  
  10.      Node  
  11.       |     ↖  
  12.       |       ↖  
  13.       P ——————> <Child>  
  14.                   |  
  15.                   a 

通过这种场景可以推测出当删除该节点时,它下面子树中的所有节点都要被删除。现在直接以这个场景为例,走一下删除过程。这个过程实际上也就是unmountHostComponents函数的运行机制。

删除过程

删除Node节点需要父DOM节点的参与:

  1. parentInstance.removeChild(child) 

所以首先要定位到父级节点。过程是在Fiber树中,以Node的父节点为起点往上找,找到的第一个原生DOM节点即为父节点。在例子中,父节点就是div。此后以Node为起点,遍历子树,子树也是fiber树,因此遍历是深度优先遍历,将每个子节点都删除。

需要特别注意的一点是,对循环节点进行删除,每个节点都会被删除操作去处理,这里的每个节点是fiber节点而不是DOM节点。DOM节点的删除时机是从Node开始遍历进行删除的时候,遇到了第一个原生DOM节点(HostComponent或HostText)这个时刻,在删除了它子树的所有fiber节点后,才会被删除。

以上是完整过程的简述,对于详细过程要明确几个关键函数的职责和调用关系才行。删除fiber节点的是unmountHostComponents函数,被删除的节点称为目标节点,它的职责为:

  1.  找到目标节点的DOM层面的父节点
  2.  判断目标节点如果是原生DOM类型的节点,那么执行3、4,否则先卸载自己之后再往下找到原生DOM类型的节点之后再执行3、4
  3.  遍历子树执行fiber节点的卸载
  4.  删除目标节点的DOM节点

其中第3步的操作,是通过commitNestedUnmounts完成的,它的职责很单一也很明确,就是遍历子树卸载节点。

然后具体到每个节点的卸载过程,由commitUnmount完成。它的职责是

  1.  Ref的卸载
  2.  类组件生命周期的调用
  3.  HostPortal类型的fiber节点递归调用unmountHostComponents重复删除过程

下面来看一下不同类型的组件它们的具体删除过程是怎样的。

区分被删除组件的类别

Node节点的类型有多种可能性,我们以最典型的三种类型(HostComponent、ClassComponent、HostPortal)为例分别说明一下删除过程。

首先执行unmountHostComponents,会向上找到DOM层面的父节点,然后根据下面的三种组件类型分别处理,我们挨个来看。

HostComponent

Node 是HostComponent,调用commitNestedUnmounts,以Node为起点,遍历子树,开始对所有子Fiber进行卸载操作,遍历的过程是深度优先遍历。 

  1. Delation   -->      Node(span)  
  2.                      |    ↖  
  3.                      |       ↖  
  4.                      P ——————> <Child>  
  5.                                  |  
  6.                                  a 

对节点逐个执行commitUnmount进行卸载,这个遍历过程其实对于三种类型的节点,都是类似的,为了节省篇幅,这里只表述一次。

Node的fiber被卸载,然后向下,p的fiber被卸载,p没有child,找到它的sibling<Child>,<Child>的fiber被卸载,向下找到a,a的fiber被卸载。此时到了整个子树的叶子节点,开始向上return。由a 到 <Child>,再回到Node,遍历卸载的过程结束。

在子树的所有fiber节点都被卸载之后,才可以安全地将Node的DOM节点从父节点中移除。

ClassComponent 

  1. Delation   -->      Node(ClassComponent)  
  2.                      |  
  3.                      |  
  4.                     span  
  5.                      |    ↖  
  6.                      |       ↖  
  7.                      P ——————> <Child>  
  8.                                  |  
  9.                                  a 

Node是ClassComponent,它没有对应的DOM节点,要先调用commitUnmount卸载它自己,之后会先往下找,找到第一个原生DOM类型的节点span,以它为起点遍历子树,确保每一个fiber节点都被卸载,之后再将span从父节点中删除。

HostPortal         

  1.    div2(Container Of Node)  
  2.            ↗  
  3. div   containerInfo  
  4.  |    ↗  
  5.  |  ↗  
  6. Node(HostPortal)  
  7.  |  
  8.  |  
  9. span  
  10.  |    ↖  
  11.  |       ↖  
  12.  P ——————> <Child>  
  13.              |  
  14.              a 

Node是HostPortal,它没有对应的DOM节点,因此删除过程和ClassComponent基本一致,不同的是删除它下面第一个子fiber的DOM节点时不是从这个被删除的HostPortal类型节点的DOM层面的父节点中删除,而是从HostPortal的containerInfo中移除,图示上为div2,因为HostPortal会将子节点渲染到父组件以外的DOM节点。

以上是三种类型节点的删除过程,这里值得注意的是,unmountHostComponents函数执行到遍历子树卸载每个节点的时候,一旦遇到HostPortal类型的子节点,会再次调用unmountHostComponents,以它为目标节点再进行它以及它子树的卸载删除操作,相当于一个递归过程。

commitUnmount

HostComponent 和 ClassComponent的删除都调用了commitUnmount,除此之外还有FunctionComponent也会调用它。它的作用对三种组件是不同的:

  •  FunctionComponent 函数组件中一旦调用了useEffect,那么它卸载的时候要去调用useEffect的销毁函数。(useLayoutEffect的销毁函数是调用commitHookEffectListUnmount执行的)
  •  ClassComponent 类组件要调用componentWillUnmount
  •  HostComponent 要卸载ref 
  1. function commitUnmount(  
  2.   finishedRoot: FiberRoot,  
  3.   current: Fiber,  
  4.   renderPriorityLevel: ReactPriorityLevel,  
  5. ): void {  
  6.   onCommitUnmount(current);  
  7.   switch (current.tag) {  
  8.     case FunctionComponent:  
  9.     case ForwardRef:  
  10.     case MemoComponent:  
  11.     case SimpleMemoComponent:  
  12.     case Block: {  
  13.       const updateQueue: FunctionComponentUpdateQueue | null = (current.updateQueue: any);  
  14.       if (updateQueue !== null) {  
  15.         const lastEffect = updateQueue.lastEffect;  
  16.         if (lastEffect !== null) {  
  17.           const firstEffect = lastEffect.next;  
  18.           let effect = firstEffect 
  19.           do { 
  20.              const {destroy, tag} = effect;  
  21.             if (destroy !== undefined) {  
  22.               if ((tag & HookPassive) !== NoHookEffect) {  
  23.                 // 向useEffect的销毁函数队列里push effect  
  24.                 enqueuePendingPassiveHookEffectUnmount(current, effect);  
  25.               } else {  
  26.                 // 尝试使用try...catch调用destroy  
  27.                 safelyCallDestroy(current, destroy);  
  28.                 ...  
  29.               }  
  30.             }  
  31.             effecteffect = effect.next;  
  32.           } while (effect !== firstEffect);  
  33.         }  
  34.       }  
  35.       return;  
  36.     }  
  37.     case ClassComponent: {  
  38.       safelyDetachRef(current);  
  39.       const instance = current.stateNode;  
  40.       // 调用componentWillUnmount  
  41.       if (typeof instance.componentWillUnmount === 'function') {  
  42.         safelyCallComponentWillUnmount(current, instance);  
  43.       }  
  44.       return;  
  45.     }  
  46.     case HostComponent: {  
  47.       // 卸载ref  
  48.       safelyDetachRef(current);  
  49.       return;  
  50.     }  
  51.     ...  
  52.   }  

总结

我们来复盘一下删除过程中的重点:

  •  删除操作执行的时机
  •  删除的目标是谁
  •  从哪里删除

mutation在基于Fiber节点对DOM做其他操作之前,需要先删除节点,保证留给后续操作的fiber节点都是有效的。删除的目标是Fiber节点及其子树和Fiber节点对应的DOM节点,整个轨迹循着fiber树,对目标节点和所有子节点都进行卸载,对目标节点对应的(或之下的第一个)DOM节点进行删除。对于原生DOM类型的节点,直接从其父DOM节点删除,对于HostPortal节点,它会把子节点渲染到外部的DOM节点,所以会从这个DOM节点中删除。明确以上三个点再结合上述梳理的过程,就可以逐渐理清删除操作的脉络。 

 

责任编辑:庞桂玉 来源: segmentfault
相关推荐

2022-07-19 13:31:18

Buddy算法内存管理框架

2020-10-07 22:21:13

程序员技术线程

2021-03-07 16:31:35

Java编译反编译

2014-06-06 16:08:17

初志科技

2011-09-19 15:40:35

2020-07-29 08:14:59

云计算云迁移IT

2018-12-26 13:22:05

NVMeNVMe-oF数据

2023-06-16 07:48:51

DOM对象JS

2017-07-19 14:26:01

前端JavaScriptDOM

2021-08-16 09:59:52

ReactSvelte开发

2012-05-31 09:53:38

2019-01-07 12:02:02

TCP长连接Java

2020-08-11 08:59:20

容器虚拟化技术

2012-05-01 08:06:49

手机

2011-05-19 16:47:50

软件测试

2018-03-01 15:03:11

2016-01-19 21:59:50

OpenStack

2017-05-15 21:50:54

Linux引号

2024-02-04 17:03:30

2015-08-20 09:17:36

Java线程池
点赞
收藏

51CTO技术栈公众号