React16源码: React中commitAllHostEffects内部的commitPlacement的源码实现

commitPlacement


1 )概述

  • 在 react commit 阶段的 commitRoot 第二个while循环中
  • 调用了 commitAllHostEffects,在这个函数内部处理了
  • 把一个新的dom节点挂载到真正的dom树上面去的一个过程
  • 现在主要关注下其中调用的 commitPlacement

2 )源码

定位到 packages/react-reconciler/src/ReactFiberCommitWork.js#L850

function commitPlacement(finishedWork: Fiber): void {if (!supportsMutation) {return;}// Recursively insert all host nodes into the parent.// 找到第一个父节点上的 HostComponent 或 HostRoot 或 HostPortalconst parentFiber = getHostParentFiber(finishedWork);// Note: these two variables *must* always be updated together.let parent;let isContainer;// 判断 tag 匹配处理程序switch (parentFiber.tag) {case HostComponent:parent = parentFiber.stateNode;isContainer = false;break;case HostRoot:parent = parentFiber.stateNode.containerInfo;isContainer = true;break;case HostPortal:parent = parentFiber.stateNode.containerInfo;isContainer = true;break;default:invariant(false,'Invalid host parent fiber. This error is likely caused by a bug ' +'in React. Please file an issue.',);}if (parentFiber.effectTag & ContentReset) {// Reset the text content of the parent before doing any insertionsresetTextContent(parent);// Clear ContentReset from the effect tagparentFiber.effectTag &= ~ContentReset;}// before可能不存在,比如如果是单一节点const before = getHostSibling(finishedWork);// We only have the top Fiber that was inserted but we need recurse down its// children to find all the terminal nodes.let node: Fiber = finishedWork;while (true) {// 只有这种情况下,才有插入dom的需要,根据 before 和 isContainer来执行不同的插入if (node.tag === HostComponent || node.tag === HostText) {if (before) {// 这里匹配 HostRoot 或 HostPortalif (isContainer) {insertInContainerBefore(parent, node.stateNode, before);} else {// 这里匹配 HostComponentinsertBefore(parent, node.stateNode, before);}} else {// 不存在 before// 匹配 HostRoot 或 HostPortalif (isContainer) {appendChildToContainer(parent, node.stateNode);} else {// 匹配 HostComponentappendChild(parent, node.stateNode);}}} else if (node.tag === HostPortal) {// If the insertion itself is a portal, then we don't want to traverse// down its children. Instead, we'll get insertions from each child in// the portal directly.} else if (node.child !== null) {// 如果都不符合,并且存在子节点,往下去找node.child.return = node;node = node.child;continue; // 拿到 child 后,跳过此次,继续while循环}// 到终点了,就结束if (node === finishedWork) {return;}while (node.sibling === null) {if (node.return === null || node.return === finishedWork) {return;}node = node.return;}node.sibling.return = node.return;node = node.sibling;}
}
  • 进入 getHostParentFiber
    function getHostParentFiber(fiber: Fiber): Fiber {let parent = fiber.return;while (parent !== null) {if (isHostParent(parent)) {return parent;}parent = parent.return;}invariant(false,'Expected to find a host parent. This error is likely caused by a bug ' +'in React. Please file an issue.',);
    }function isHostParent(fiber: Fiber): boolean {return (fiber.tag === HostComponent ||fiber.tag === HostRoot ||fiber.tag === HostPortal);
    }
    
    • 显然意见,这个循环就是向上查找到第一个 HostComponentHostRootHostPortal
  • 之后,判断 parentFiber.tag,对不同条件的 isContainer 进行赋值
  • 之后,判断 ContentReset 是否存在,存在则对文本节点进行替换
  • 进入 getHostSibling 向上找到 sibling 节点,下面这个英文注释留着 这个是查找sibling的核心算法
    function getHostSibling(fiber: Fiber): ?Instance {// We're going to search forward into the tree until we find a sibling host// node. Unfortunately, if multiple insertions are done in a row we have to// search past them. This leads to exponential search for the next sibling.// TODO: Find a more efficient way to do this.let node: Fiber = fiber;// 这里定义一个while循环叫做 siblingssiblings: while (true) {// If we didn't find anything, let's try the next sibling.// 没有兄弟节点while (node.sibling === null) {if (node.return === null || isHostParent(node.return)) {// If we pop out of the root or hit the parent the fiber we are the// last sibling.return null;}node = node.return; // 向父级溯源}node.sibling.return = node.return; // 这里本就相同node = node.sibling; // 兄弟节点变成当前// 这个循环要找兄弟节点中的第一个dom节点// 如果兄弟节点不是 HostComponent 或 HostText 往下去找// 子节点中的第一个dom节点while (node.tag !== HostComponent && node.tag !== HostText) {// If it is not host node and, we might have a host node inside it.// Try to search down until we find one.// 当前是要替换的节点,就没有必要向下找了if (node.effectTag & Placement) {// If we don't have a child, try the siblings instead.continue siblings;}// If we don't have a child, try the siblings instead.// We also skip portals because they are not part of this host tree.// 像是这种,也没有必要继续找了               if (node.child === null || node.tag === HostPortal) {continue siblings;} else {node.child.return = node;node = node.child;}}// Check if this host node is stable or about to be placed.if (!(node.effectTag & Placement)) {// Found it!return node.stateNode;}}
    }
    
  • 接下去又进入一个while循环
    • 里面的第一个判断,node.tag === HostComponent || node.tag === HostText
    • 只有 HostComponent 和 HostText 才有插入dom的需要
    • 注意,这里一系列的 if-else 是操作dom的核心
  • react真正展现给用户的是一棵dom树,而react中存储的是fiber树
  • 而fiber树不会有每个节点对应的dom节点
  • 以上是操作 commitPlacement 的源码处理,主要关注的是上述while循环和判断

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/651079.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

软考15-上午题-编译程序基本原理

一、编译过程【回顾】 中间代码生成、代码优化,可省略。 二、正规式 词法分析的工具。 ab*:这个*针对的是b,即b可以出现0次或多次。 2-1、真题 真题1: 真题2: 真题3: 真题4: 真题5&#xff1a…

Python网络爬虫实战——实验7:Python使用apscheduler定时采集任务实战

【实验内容】 本实验主要介绍在Django框架中使用APScheduler第三方库实现对数据的定时采集。 【实验目的】 1、掌握APScheduler库的使用; 2、学习在Django中实现多个定时任务调度; 【实验步骤】 步骤1 Apscheduler简介与特点 步骤2 Apscheduler基本…

Linux进程间通信(IPC)机制之一:管道(Pipes)详解

🎬慕斯主页:修仙—别有洞天 ♈️今日夜电波:Nonsense—Sabrina Carpenter 0:50━━━━━━️💟──────── 2:43 🔄 ◀️ ⏸ ▶️ …

Scrapy IP()类 编程指南(基础)

Scrapy IP()类 编程指南(基础) IP简介 工欲善其事,必先利其器,在聊Scapy IP类时,我们先要了解IP是什么。 IP指的是Internet Protocol(互联网协议)的数据包。Internet Protocol是互联网上用于在…

SpringBoot系列之JPA实现按年月日查询

SpringBoot系列之JPA实现按年月日查询 通过例子的方式介绍Springboot集成Spring Data JPA的方法,进行实验,要先创建一个Initializer工程,如图: 选择,需要的jdk版本,maven项目 选择需要的maven配置&#x…

pytest参数化

一、pytest.mark.parametrize介绍 pytest.mark.parametrize(argnames, argvalues, indirectFalse, idsNone)参数说明: argnames: 一个或多个参数名,用逗号分隔的字符串,如"arg1,arg2,arg3",参数名与用例入参数一致。 a…

单元测试——题目十三

目录 题目要求: 定义类 测试类 题目要求: 根据输入的三条边值判断能组成何种三角形。三条边为变量a、b、c,范围为1≤边值≤10,不在范围内,提示“输入边值不在范围内”。不满足任意两边之和必须大于第三边,提示“输入边值不能组成三角形”。输入边值能组成三角形,只有…

Android读写文件,适配Q以上

Android Q升级了文件系统,访问文件不仅仅是说动态权限了,有各种限制。权限什么的就不赘述了,下面介绍一下在10以上的系统中访问文件。 首先是打开文件管理器 /*** 打开文件管理器 存储卡和外接U盘都可以访问*/public void openFileManager()…

什么是JSON

什么是JSON JSON:JavaScript Object Notation 【JavaScript 对象表示法】 JSON 是存储和交换文本信息的语法。类似 XML。 JSON采用完全独立于任何程序语言的文本格式,使JSON成为理想的数据交换语言S 为什么需要JSON 提到JSON,我们就应该和…

代码块(Java)

代码块是类的成分之一,分为静态代码块和实例代码块 1.静态代码块:static{} 类加载时会自动执行一次,可以完成类的初始化,比如初始化赋值 2.实例代码块:{} 每次创建对象时,执行实例代码块,会…

探索圆的面积计算器:神秘数学背后的无限魅力

导语:自古以来,圆形在人类文明中扮演着重要角色。从建筑、工程设计到日常生活,圆的面积都是一个不可或缺的元素。本文将带您深入了解圆的面积,从起源、应用场景到计算方法,让您领略数学的无穷魅力。 一、圆的面积是什…

java金额数字转中文

java金额数字转中文 运行结果: 会进行金额的四舍五入。 工具类源代码: /*** 金额数字转为中文*/ public class NumberToCN {/*** 汉语中数字大写*/private static final String[] CN_UPPER_NUMBER {"零", "壹", "贰",…

CMake构建Qt工程

在https://blog.csdn.net/fengbingchun/category_12172633.html 上有直接通过vs2022建的Console、Widgets、Quick三个工程,这里增加通过CMake构建。 build.sh内容如下: #! /bin/bashif [ $# ! 1 ]; thenecho "Error: requires one parameter: Rele…

基于springboot网上书城交易平台源码和论文

在Internet高速发展的今天,我们生活的各个领域都涉及到计算机的应用,其中包括网上书城管理系统的网络应用,在国外网上书城管理系统已经是很普遍的方式,不过国内的书城管理系统可能还处于起步阶段。网上书城管理系统具有网上书城信…

Spring扩展点在微服务应用(待完善)

ApplicationListener扩展 nacos注册服务, 监听容器发布事件 # 容器发布事件 AbstractAutoServiceRegistration#onApplicationEvent # 接收事件吗,注册服务到nacos NacosServiceRegistry#register Lifecycle扩展 #订阅服务实例更改的事件 NamingService#…

MySQL数据库的一些缩写含义

DDL Data Definition Language,数据定义语言,用来定义数据库对象(数据库,表,字段) DML DML英文全称是Data Manipulation Language(数据操作语言),用来对数据库中表的数据记录进 行增、删、改操作。 添加数据&#x…

【开源】基于JAVA的学生综合素质评价系统

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 学生功能2.2 教师功能2.3 教务处功能 三、系统展示四、核心代码4.1 查询我的学科竞赛4.2 保存单个问卷4.3 根据类型查询学生问卷4.4 填写语数外评价4.5 填写品德自评问卷分 五、免责说明 一、摘要 1.1 项目介绍 基于J…

24校招,经纬恒润测试工程师PPT技术二面

前言 大家好,我是chowley,今天回顾一下之前参加的PPT工程师面试(doge) 面试官针对我的PPT进行提问,梦回大学课堂 时间:40min 平台:腾讯会议 过程 20minPPT展示PPT做了多长时间&#xff1f…

【数据结构】栈、队列、数组、列表

数据结构是什么? 数据结构是计算机存储、组织数据的方式 是指数据相互之间是以什么方式排列在一起的。 数据结构是为了更加方便的管理和使用数据,需要结合具体的业务场景来进行选择。一般情况下,精心选择的数据结构可以带来更高的运行或者…

MongoDB日期存储与查询、@Query、嵌套字段查询实战总结

缘由 MongoDB数据库如下: 如上截图,使用MongoDB客户端工具DataGrip,在filter过滤框输入{ profiles.alias: 逆天子, profiles.channel: },即可实现昵称和渠道多个嵌套字段过滤查询。 现有业务需求:用Java代码来查询…