iis服务器助手广告广告
返回顶部
首页 > 资讯 > 前端开发 > node.js >如何使用React和DOM的节点删除算法
  • 180
分享到

如何使用React和DOM的节点删除算法

2024-04-02 19:04:59 180人浏览 薄情痞子
摘要

本篇内容主要讲解“如何使用React和DOM的节点删除算法”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“如何使用React和DOM的节点删除算法”吧!Fiber

本篇内容主要讲解“如何使用React和DOM的节点删除算法”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“如何使用React和DOM的节点删除算法”吧!

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

function commitMutationEffects(    firstChild: Fiber,    root: FiberRoot,    renderPriorityLevel,  ) {    let fiber = firstChild;    while (fiber !== null) {      // 首先进行删除      const deletions = fiber.deletions;      if (deletions !== null) {        commitMutationEffectsDeletions(deletions, root, renderPriorityLevel);      }      // 如果删除之后的fiber还有子节点,      // 递归调用commitMutationEffects来处理      if (fiber.child !== null) {        const primarySubtreeTag = fiber.subtreeTag & MutationSubtreeTag;        if (primarySubtreeTag !== NoSubtreeTag) {          commitMutationEffects(fiber.child, root, renderPriorityLevel);        }      }      if (__DEV__) {} else {        // 执行其他DOM操作        try {          commitMutationEffectsImpl(fiber, root, renderPriorityLevel);        } catch (error) {          captureCommitPhaseError(fiber, error);        }      }      fiberfiber = fiber.sibling;    }  }

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

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

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

Fiber树     div#root        |      <App/>        |       div        |     <Parent/>        |       Node        |     ↖        |       ↖        P &mdash;&mdash;&mdash;&mdash;&mdash;&mdash;> <Child>                    |                    a

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

删除过程

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

parentInstance.removeChild(child)

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

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

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

  1. 鸿蒙官方战略合作共建——HarmonyOS技术社区

  2.  找到目标节点的DOM层面的父节点

  3.  判断目标节点如果是原生DOM类型的节点,那么执行3、4,否则先卸载自己之后再往下找到原生DOM类型的节点之后再执行3、4

  4.  遍历子树执行fiber节点的卸载

  5.  删除目标节点的DOM节点

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

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

  1. 鸿蒙官方战略合作共建——HarmonyOS技术社区

  2.  Ref的卸载

  3.  类组件生命周期的调用

  4.  HostPortal类型的fiber节点递归调用unmountHostComponents重复删除过程

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

区分被删除组件的类别

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

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

HostComponent

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

Delation   -->      Node(span)                       |    ↖                       |       ↖                       P &mdash;&mdash;&mdash;&mdash;&mdash;&mdash;> <Child>                                   |                                   a

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

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

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

ClassComponent

Delation   -->      Node(ClassComponent)                       |                       |                      span                       |    ↖                       |       ↖                       P &mdash;&mdash;&mdash;&mdash;&mdash;&mdash;> <Child>                                   |                                   a

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

HostPortal       

   div2(Container Of Node)             ↗  div   containerInfo   |    ↗   |  ↗  Node(HostPortal)   |   |  span   |    ↖   |       ↖   P &mdash;&mdash;&mdash;&mdash;&mdash;&mdash;> <Child>               |               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 

function commitUnmount(    finishedRoot: FiberRoot,    current: Fiber,    renderPriorityLevel: ReactPriorityLevel,  ): void {    onCommitUnmount(current);    switch (current.tag) {      case FunctionComponent:      case ForwardRef:      case MemoComponent:      case SimpleMemoComponent:      case Block: {        const updateQueue: FunctionComponentUpdateQueue | null = (current.updateQueue: any);        if (updateQueue !== null) {          const lastEffect = updateQueue.lastEffect;          if (lastEffect !== null) {            const firstEffect = lastEffect.next;            let effect = firstEffect;            do {              const {destroy, tag} = effect;              if (destroy !== undefined) {                if ((tag & HookPassive) !== NoHookEffect) {                  // 向useEffect的销毁函数队列里push effect                  enqueuePendingPassiveHookEffectUnmount(current, effect);                } else {                  // 尝试使用try...catch调用destroy                  safelyCallDestroy(current, destroy);                  ...                }              }              effecteffect = effect.next;            } while (effect !== firstEffect);          }        }        return;      }      case ClassComponent: {        safelyDetachRef(current);        const instance = current.stateNode;        // 调用componentWillUnmount        if (typeof instance.componentWillUnmount === 'function') {          safelyCallComponentWillUnmount(current, instance);        }        return;      }      case HostComponent: {        // 卸载ref        safelyDetachRef(current);        return;      }      ...    }  }

总结

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

  •  删除操作执行的时机

  •  删除的目标是谁

  •  从哪里删除

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

到此,相信大家对“如何使用React和DOM的节点删除算法”有了更深的了解,不妨来实际操作一番吧!这里是编程网网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!

--结束END--

本文标题: 如何使用React和DOM的节点删除算法

本文链接: https://www.lsjlt.com/news/83940.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 如何使用React和DOM的节点删除算法
    本篇内容主要讲解“如何使用React和DOM的节点删除算法”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“如何使用React和DOM的节点删除算法”吧!Fiber...
    99+
    2024-04-02
  • js中怎么使用removeChild()方法删除dom节点
    这篇文章将为大家详细讲解有关js中怎么使用removeChild()方法删除dom节点,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。JavaScript是什么JavaScript是一种直译式的脚本语言,其...
    99+
    2023-06-14
  • JavaScript中dom如何添加、删除节点
    这篇文章将为大家详细讲解有关JavaScript中dom如何添加、删除节点,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。javascript是一种什么语言javascript是一种动态类型、弱类型的语言,...
    99+
    2023-06-14
  • 在JavaScript的jQuery中如何使用detach()方法删除dom节点元素
    这篇文章主要介绍在JavaScript的jQuery中如何使用detach()方法删除dom节点元素,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!JavaScript有什么特点1、js属于一种解释性脚本语言;2、在绝...
    99+
    2023-06-14
  • 怎么在jQuery中使用remove()方法删除dom节点
    这篇文章将为大家详细讲解有关怎么在jQuery中使用remove()方法删除dom节点,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。jquery是什么jquery是一个简洁而快速的JavaS...
    99+
    2023-06-14
  • 怎么在JavaScript中使用remove方法删除dom节点
    今天就跟大家聊聊有关怎么在JavaScript中使用remove方法删除dom节点,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。javascript是一种什么语言javascript...
    99+
    2023-06-14
  • angular中如何动态删除ng-repaeat添加的dom节点
    这篇文章主要为大家展示了“angular中如何动态删除ng-repaeat添加的dom节点”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“angular中如何动态...
    99+
    2024-04-02
  • 在jQuery库中dom节点删除方法detach()和remove()有哪些区别
    这篇文章给大家分享的是有关在jQuery库中dom节点删除方法detach()和remove()有哪些区别的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。jquery是什么jquery是一个简洁而快速的JavaSc...
    99+
    2023-06-14
  • 如何删除redis集群的节点
    删除redis集群节点的方法:可以通过del-node命令进行删除从节点即可,例如:cd /usr/local/redis/redis/src./redis-trib.rb del-node 192.168.182.132:7007 //7...
    99+
    2024-04-02
  • 如何删除Linked List中的节点
    今天就跟大家聊聊有关如何删除Linked List中的节点,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。Delete Node in a Linked List题目请编写一个函数,使...
    99+
    2023-06-19
  • ceph中如何删除和添加osd节点
    这篇文章主要介绍了ceph中如何删除和添加osd节点,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。在ceph中,数据是以PG为单位保存在osd中的,一般情况下一个osd会对应...
    99+
    2023-06-04
  • 如何使用JavaScript永久删除dom元素
    今天小编给大家分享一下如何使用JavaScript永久删除dom元素的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。解决方案假...
    99+
    2023-06-26
  • javascript如何删除当前元素的父节点
    这篇文章主要介绍javascript如何删除当前元素的父节点,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完! javascript删除父节点的方法:1、通过...
    99+
    2024-04-02
  • MongoDB副本集如何添加和删除仲裁节点
    小编给大家分享一下MongoDB副本集如何添加和删除仲裁节点,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧! ...
    99+
    2024-04-02
  • Oracle添加和删除集群节点的方法是什么
    这篇文章主要介绍“Oracle添加和删除集群节点的方法是什么”,在日常操作中,相信很多人在Oracle添加和删除集群节点的方法是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解...
    99+
    2024-04-02
  • C++链表节点的添加和删除方法是什么
    本篇内容介绍了“C++链表节点的添加和删除方法是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!前言链表是一种动态的数据结构,因为在创建链...
    99+
    2023-06-26
  • CSS上下文选择器与使用DOM节点的方法
    本篇内容主要讲解“CSS上下文选择器与使用DOM节点的方法”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“CSS上下文选择器与使用DOM节点的方法”吧!   S...
    99+
    2024-04-02
  • jquery如何删除指定元素后的兄弟节点
    这篇文章主要为大家展示了“jquery如何删除指定元素后的兄弟节点”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“jquery如何删除指定元素后的兄弟节点”这篇文...
    99+
    2024-04-02
  • JavaScript节点的增删改查方法怎么使用
    本篇内容主要讲解“JavaScript节点的增删改查方法怎么使用”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“JavaScript节点的增删改查方法怎么使用”吧!节点的增删改查节点的创建docu...
    99+
    2023-07-05
  • 使用递归怎么删除树形结构的所有子节点
    使用递归怎么删除树形结构的所有子节点?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。1.业务场景有如下树形结构: +—0 +—1 +—2 +—4 +—5 +—3如...
    99+
    2023-05-31
    递归
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作