自顶而下学习react源码 (2)架构篇 render阶段

架构

Reconciler工作的阶段属于render阶段,Renderer工作的阶段属于commit阶段。

rendercommit阶段统称为work,即React在工作中。相对应的,如果任务正在Scheduler内调度,就不属于work

render阶段 分为递归两个阶段。

递阶段

递阶段是深度优先遍历,为遍历到的每个fiber节点调用beginWork方法。该方法会根据传入的fiber节点创建子fibei节点,并且将fiber节点连接起来(不会创建dom)。当遍历到叶子节点的时候,该叶子节点完成递阶段的同时也开始归阶段。

归阶段

归阶段会调用completework处理fiber节点。当某个节点完成了completework方法,就表示该节点工作完毕,到下一个节点。顺序就是:找兄弟节点,如果有兄弟节点,就让他进入递阶段,若没有兄弟节点,就会进入父fiber的归阶段。递归阶段会交错执行知道rootfiber完成归阶段,那么render阶段就算完成了。

例子:

function App() {
  return (
    <div>
      i am
      <span>aa</span>
    </div>
  )
}

ReactDOM.render(<App />, document.getElementById("root"));

对应的fiber树: rootfiber => App => div =>. i am

​ => span => aa

那么render阶段的顺序就是:

1 Rootfiber beginwork
2 App beginwork
3 div begingwork
4 "i am"beginwork // iam属于第一个叶子节点
5  i am completework
6 span beginwork
7 span completework
8 div completework
9 app completework
10 rootfiber completework

aa没有beginwork/completework,作为一种性能优化手段,针对只有单一文本节点的fiber,会特殊处理。

beginwork

在这里插入图片描述
入参三个,current是当前fiber节点对应上次更新时候的fiber节点,通常通过current 是否等于null来判断当前是mount和update。

update时,如果current存在,满足一定条件可以服用current节点,
在这里插入图片描述
对于常见的类型,如FunctionComponent或者ClassComponent,最终都会进入reconcilerChildren方法。

function updateHostRoot(
  // 当前组件对应的Fiber节点在上一次更新时的Fiber节点,即workInProgress.alternate
  current: Fiber | null,
  //mount的时候,除了rootfiber,其他fiber没有上一次更新的对应的fiber,即current = null
  // 当前组件对应的Fiber节点
  workInProgress: Fiber,
  // 优先级相关,在讲解Scheduler时再讲解
  renderLanes: Lanes,) {
   // 省略 
   reconcileChildren(current, workInProgress, nextChildren, renderLanes);
   //省略
  // beginWork返回值,并作为下次performUnitOfWork执行时workInProgress的传参,一个工作节点就“递”完毕了
  return workInProgress.child;
  }
reconcileChildren

Reconciler模块的核心部分,它主要完成两件事情。

  • 1 对于mount组件,他会创建新的子fiber节点,调用mountChildFibers方法,但不会为fiber打上effectTag属性。
  • 2 对于update组件,他会将当前组件与上次的fiber节点比较(diff算法),将新的结果生成新的fiber节点。打上标记,如update。
    在这里插入图片描述
effectTag

render阶段是在内存中进行的,当工作结束后会通知Renderer模块需要执行的dom操作,而如何操作的类型就保存在fiber.effetTag中。

// DOM需要插入到页面中
export const Placement = /*                */ 0b00000000000010;
// DOM需要更新
export const Update = /*                   */ 0b00000000000100;
// DOM需要插入到页面中并更新
export const PlacementAndUpdate = /*       */ 0b00000000000110;
// DOM需要删除
export const Deletion = /*                 */ 0b00000000001000;

如果想将fiber节点对应的dom插入页面中,需要满足:

1 fiber.stateNode存在,即fiber节点保存了对应的DOM节点 。

2 fiber.effectTag === Placement,即fiber节点存在Placement effectTag。

第一次beginWork的时候,fiber.stateNode === null,而且mountChildFibers不会为fiber节点赋值effectTag,所以fiber.stateNode会在归阶段,completeWork创建。

而第二个问题,在第一次递归阶段,整颗fiber树只有root fiber会复制Placement effectTag,而其他fiber节点不会,不然需要大量的操作DOM,非常低效。

CompleteWork

在这里插入图片描述
completeWork的入参根beginWork一样,处理逻辑主要是根据当前fiber节点的tag做不同出炉,这里需要重点观看HostComponent的处理,

对于HostComponent,当处于update阶段的时候,调用updateHostComponent对fiber的props做一些处理。
在这里插入图片描述
updateHostComponent内部,被处理完的props会赋值给updatePayload,最终赋值到fiber.updateQueue上,(workInProgress就是当前的fiber),然后在commit阶段的时候渲染到页面上。

而对于mount阶段,主要逻辑包括三个。

1 为当前fiber生成对应的dom节点,

2 将子孙dom插入刚生成的dom节点中

3 与update逻辑中的updateHostComponent类似的处理props的过程。
在这里插入图片描述
上面说过mount的时候,只有root fiber上存在Placement effectTag,那么commit阶段是如何一次插入整颗dom树的,原因就是completeWork的appendAllChildren方法!!!因为completeWork是在归阶段完成的,按照递归阶段的顺序,叶子节点会先完成,递归阶段,然后再一步一步往上。比如

例子:

function App() {
  return (
    <div>
      i am
      <span>aa</span>
    </div>
  )
}

ReactDOM.render(<App />, document.getElementById("root"));

对应的fiber树: rootfiber => App => div =>. i am

​ => span => aa

1 Rootfiber beginwork
2 App beginwork
3 div begingwork
4 "i am"beginwork // iam属于第一个叶子节点
5  i am completework
6 span beginwork
7 span completework
8 div completework
9 app completework
10 rootfiber completework

到 div fiber执行completework的时候,i am和span的fiber都已经完成了completeWork方法,也都创建了对应的dom,所以就可以通过appendAllCHildren将他们的dom插入到div下,以此类推,到rootFiber执行completeWork的时候,我们已经有一颗完整的dom树了。然后再commit阶段直接插入一次即可。

总结: Reconciler阶段称为render阶段,它主要分为两个步骤,递归阶段,递阶段每个fiber节点执行beginWork方法。他会为子节点创建fiber并且连接起来,但不会创建dom,也不会为fiber节点打上effectTag标记,(除了update的时候,update的时候bieginWork会为fiber节点打上变化的标记。)接着再一步一步归回来,归阶段执行completeWork方法,在update的时候该方法主要会处理props,在mount的时候该方法主要会创建dom节点,并且将子孙dom节点插入到当前的节点上,由于是从下往上归上来的,所以等到了root fiber执行completeWork的时候,就已经拥有一颗完整的dom树了。
递阶段是深度优先遍历,当当前的fiber没有子节点的时候,执行完递阶段就会执行归阶段,然后找兄弟节点,执行递归阶段,当没有兄弟节点的时候,再往回执行父节点的归阶段,然后一步一步归回去,知道rootfiber执行完completeWork,render阶段就算结束。

effectList

render阶段也就是Recomciler阶段,会为需要更新的fiber打上effectTag标记,难道commit阶段还需要遍历一次fiber树,获取那些effectTag不为null的fiber节点吗?答案是否定的。

为了解决这个问题,在completeWork的上层函数completeUnitWork中,每个执行完completeWork且存在effectTag的fiber会被保存在一个称为effectList的单向链表中,

在归阶段中,所有有effectTag的f iber节点都会追加到一条链表中,最终形成一个以rootFiber.firstEfect为起点的链表。

rootFiber.firstEffect -----nextEffect------> fiber ------nextEffect-----> fiber

在commit阶段只需要遍历该链表即可。effectList相对于fiber树而言,就像是圣诞树上的一串彩灯。

至此,render阶段全部完毕,在performSyncWorkOnRoot函数中。fiberRootNode被传递给commitRoot方法,开启了commit阶段。
在这里插入图片描述

学习文章来自: https://react.iamkasong.com/

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
### 回答1: 学习 React 码是一项很有益的学习活动,可以帮助你加深对 React 的理解,并且为你提供有关如何实现组件化和声明式编程的知识。 在学习 React 码时,我有几个建议: 1. 先从官方文档入手,了解 React 的基本概念和使用方法。这将有助于你更好地理解 React 码。 2. 认真阅读码注释。React码非常详细,并且有很多注释,这些注释很有帮助,可以帮助你理解码的目的和工作原理。 3. 使用调试工具,以帮助你更好地理解码的工作流程。比如,可以使用浏览器的开发者工具调试 React 代码,或者使用断点调试。 4. 尝试实现一些自己的功能,并结合码来调试和理解。这将有助于你更好地理解 React 的工作原理,并且能够让你在实际使用中更加熟练。 5. 可以尝试加入 React社区,与其他 React 开发者交流经验,也可以参与 React 的开项目,为 React 贡献代码。这将有助于你不断学习和提升。 ### 回答2: 学习React码是一个非常有挑战性但也非常有价值的任务。首先,我建议你先熟悉React的基本概念和使用方法,理解React的核心思想和设计理念,然后再深入研究码。以下是一些具体的建议: 1. 掌握JavaScript和ES6的基础知识:React码是用JavaScript编写的,因此对于JavaScript的理解是必要的。确保你对JavaScript中ES6的新特性例如模块导入导出、箭头函数、解构赋值等有一定的了解。 2. 阅读官方文档和码注释:React官方提供了非常详细的文档和码注释,阅读它们可以帮助你快速了解React的内部工作原理和实现细节。 3. 从层开始分析:从React层入手,逐步深入码。首先阅读React的入口文件,了解React是如何初始化和渲染组件的。然后再深入了解React组件的生命周期和更新机制。 4. 使用工具进行调试:使用Chrome DevTools等工具进行调试可以帮助你更好地理解和分析React的运行时行为。通过在码中设置断点并观察变量的值,可以深入理解React的执行流程。 5. 理解虚拟DOM和协调算法:React的核心是虚拟DOM和协调算法,学习和理解它们对于深入理解React码非常重要。阅读相关的码实现,将有助于你理解React如何高效地更新DOM。 6. 参考社区资和开项目:React有一个非常庞大的社区,许多优秀的开项目可以帮助你更好地理解React码。与其他人一起学习和讨论,参与到React相关的开项目中,将加快你的学习进度。 总结起来,学习React码需要坚实的JavaScript基础和对React的理解。通过仔细阅读官方文档、码注释和调试工具的使用,以及参考社区资和开项目,你将能够逐渐深入了解React的实现细节,提高自己的技术水平。 ### 回答3: 学习React码的过程可以相当艰巨,但也是丰富而值得的。以下是几些建议帮助你更好地学习React码: 1. 先理解React的核心概念:在深入研究码前,先确保你对React的基本理念和工作方式有了清晰的理解。了解虚拟DOM、组件生命周期、状态管理和数据流等核心概念将有助于更好地理解码。 2. 创建一个简单的React应用:在学习React码时,建议你尝试创建一个简单的React应用并深入研究它的实现细节。这样可以帮助你更好地理解React的工作原理和内部机制。 3. 阅读React官方文档和码注释:React码中有丰富的注释,它们会对码的实现细节进行解释和说明。同时,React官方文档也提供了很多有价值的内容,包括API文档和设计原则。阅读这些内容能够帮助你更好地理解React的思想和设计理念。 4. 利用调试工具:学习码的过程中,调试工具是非常有用的辅助资。使用断点和调试器来观察码的执行路径和状态变化,可以帮助你更好地理解代码的运行机制。 5. 参考优秀的码解析和开项目:很多开发者在学习React码后,会产生一些优秀的博客文章、视频教程和开项目,其中包含了他们对码的解析和实践。阅读这些资可以帮助你更好地理解React码并从中获取灵感。 学习React码需要耐心和持续努力,希望以上建议能够帮助你更好地进行这个过程。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

coderlin_

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值