forked from elecrabbit/front-end-interview
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathfiber.md
437 lines (323 loc) · 32.7 KB
/
fiber.md
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
# React Fiber 架构解析
点击关注本[公众号](#公众号)获取文档最新更新,并可以领取配套于本指南的 **《前端面试手册》** 以及**最标准的简历模板**.
---
本文来源于[Max Koretskyi aka Wizard](https://medium.com/@maxim.koretskyi)
的[Inside Fiber: in-depth overview of the new reconciliation algorithm in React
](https://medium.com/react-in-depth/inside-fiber-in-depth-overview-of-the-new-reconciliation-algorithm-in-react-e1c04700ef6e)
向更深入理解React的全貌,建议阅读此[仓库](https://github.com/pomber/didact),它会交给你如何打造一个简易的react
---
React 是一个用于构建用户交互界面的 JavaScript 库,其核心机制就是跟踪组件的状态变化,并将更新的状态映射到到新的界面。在 React 中,我们将此过程称之为**协调**。我们调用 `setState`方法来改变状态,而框架本身会去检查 state 或 props 是否已经更改来决定是否重新渲染组件。<br />React 的官方文档对 [协调机制](https://reactjs.org/docs/reconciliation.html) 进行了良好的抽象描述: React 的元素、生命周期、 `render` 方法,以及应用于组件子元素的 diffing 算法综合起到的作用,就是协调。从 `render` 方法返回的不可变的 React 元素通常称为「虚拟 DOM」。这个术语有助于早期向人们解释 React,但它也引起了混乱,并且不再用于 React 文档。在本文中,我将坚持称它为 React 元素的树。<br />除了 React 元素的树之外,框架总是在内部维护一个实例来持有状态(如组件、 DOM 节点等)。从版本 16 开始, React 推出了内部实例树的新的实现方法,以及被称之为 **Fiber** 的算法。如果想要了解 Fiber 架构带来的优势,可以看下 [React 在 Fiber 中使用链表的方式和原因](https://medium.com/react-in-depth/the-how-and-why-on-reacts-usage-of-linked-list-in-fiber-67f1014d0eb7)。
这是本系列的第一篇文章,这一系列的目的就是向你描绘出 React 的内部架构。在本文中,我希望能够提供一些与算法相关的重要概念和数据结构,并对其进行深入阐述。一旦我们有足够的背景,我们将探索用于遍历和处理 Fiber 树的算法和主要功能。本系列的下一篇文章将演示 React 如何使用该算法执行初始渲染和处理 state 以及 props 的更新。到那时,我们将继续讨论调度程序的详细信息,子协调过程以及构建 effect 列表的机制。<br />我将给你带来一些非常高阶的知识🧙。我鼓励你阅读来了解 Concurrent React 的内部工作的魔力。如果您计划开始为 React 贡献代码,本系列也将为您提供很好的指导。我是 [逆向工程的坚定信徒](https://blog.angularindepth.com/level-up-your-reverse-engineering-skills-8f910ae10630),因此本文会有很多最新版本 16.6.0 中的源代码的链接。<br />需要消化的内容绝对是很多的,所以如果你当下还不能很理解的话,不用感到压力。花些时间是值得的。**请注意,只是使用 React 的话,您不需要知道任何文中的内容。本文是关于 React 在内部是如何工作的。
## 背景介绍
如下是我将在整个系列中使用的一个简单的应用程序。我们有一个按钮,点击它将会使屏幕上渲染的数字加 1:<br />![](https://cdn.nlark.com/yuque/0/2019/gif/128853/1564754956577-e0ebe7ee-af11-4e2e-8712-2d5eb1b986bb.gif#align=left&display=inline&height=68&originHeight=68&originWidth=210&size=0&status=done&width=210)<br />而它的实现如下:
```jsx
class ClickCounter extends React.Component {
constructor(props) {
super(props);
this.state = {count: 0};
this.handleClick = this.handleClick.bind(this);
}
handleClick() {
this.setState((state) => {
return {count: state.count + 1};
});
}
render() {
return [
<button key="1" onClick={this.handleClick}>Update counter</button>,
<span key="2">{this.state.count}</span>
]
}
}
```
你可以在 [这里](https://stackblitz.com/edit/react-t4rdmh) 把玩一下。如您所见,它就是一个可从 `render` 方法返回两个子元素 -- `button` 和 `span` 的简单组件。只要你单击该按钮,组件的状态将在处理程序内被更新,而状态的更新就会导致 `span` 元素内的文本更新。<br />在协调阶段内,React 进行了各种各样的活动。例如,在我们的简单应用程序中,从第一次渲染到状态更新后的期间内,React 执行了如下高阶操作:
- 更新了 `ClickCounter` 组件的内部状态的 `count` 属性<br />
- 获取和比较了 `ClickCounter` 组件的子组件及其 props<br />
- 更新 `span` 元素的 props<br />
- 更新 `span` 元素的 `textContent` 属性<br />
除了上述活动,React 在协调期间还执行了一些其他活动,如调用生命周期方法或更新 refs。**所有这些活动在 Fiber 架构中统称为「工作」。**工作类型通常取决于 React 元素的类型。例如,对于类定义的组件,React 需要创建实例,但是函数定义的组件就不必执行此操作。正如我们所了解的,React 中有许多元素类型,例如:类和函数组件,宿主组件(DOM 节点)portal 等。React 元素的类型由 createElement函数的第一个参数定义,此函数通常在 `render` 方法中调用以创建元素。<br />在我们开始探索活动细节和 Fiber 算法的主要内容之前,我们首先来熟悉下 React 在内部使用的一些数据结构。
## 从 React 元素到 Fiber 节点
React 中的每个组件都有一个 UI 表示,我们可以称之为从 `render` 方法返回的一个视图或模板。这是 `ClickCounter` 组件的模板:
```jsx
<button key="1" onClick={this.onClick}>Update counter</button>
<span key="2">{this.state.count}</span>
```
### React 元素
如果模板经过 JSX 编译器处理,你就会得到一堆 React 元素。这是从 React 组件的 render 方法返回的,但并不是 HTML 。由于我们并没有被强制要求使用 JSX,因此我们的 ClickCounter 组件的 render 方法可以像这样重写:
```jsx
class ClickCounter {
...
render() {
return [
React.createElement(
'button',
{
key: '1',
onClick: this.onClick
},
'Update counter'
),
React.createElement(
'span',
{
key: '2'
},
this.state.count
)
]
}
}
```
`render` 方法中调用的 `React.createElement` 会产生两个如下的数据结构:
```jsx
[
{
$$typeof: Symbol(react.element),
type: 'button',
key: "1",
props: {
children: 'Update counter',
onClick: () => { ... }
}
},
{
$$typeof: Symbol(react.element),
type: 'span',
key: "2",
props: {
children: 0
}
}
]
```
可以看到,React 为这些对象添加了 `$$typeof` 属性,从而将它们唯一地标识为 React 元素。此外我们还有属性 `type`、`key` 和 `props` 来描述元素。这些值取自你传递给 `React.createElement`函数的参数。请注意React 如何将文本内容表示为 `span` 和 `button` 节点的子项,以及 click 钩子如何成为 `button` 元素 props 的一部分。 React 元素上还有其他字段,如 `ref` 字段,而这超出了本文的范围。<br />而 `ClickCounter` 的 React 元素就没有什么 props 或 key 属性:
```jsx
{
$$typeof: Symbol(react.element),
key: null,
props: {},
ref: null,
type: ClickCounter
}
```
### Fiber 节点
在`协调`期间,从 `render` 方法返回的每个 React 元素的数据都会被合并到 Fiber 节点树中。每个 React 元素都有一个相应的 Fiber 节点。与 React 元素不同,不会在每次渲染时重新创建这些 Fiber 。这些是持有组件状态和 DOM 的可变数据结构。<br />我们之前讨论过,根据不同 React 元素的类型,框架需要执行不同的活动。在我们的示例应用程序中,对于类组件 `ClickCounter` ,它调用生命周期方法和 `render` 方法,而对于 `span` 宿主组件(DOM 节点),它进行得是 DOM 修改。因此,每个 React 元素都会转换为 [相应类型](https://github.com/facebook/react/blob/769b1f270e1251d9dbdce0fcbd9e92e502d059b8/packages/shared/ReactWorkTags.js) 的 Fiber 节点,用于描述需要完成的工作。<br />**您可以将 Fiber 视为表示某些要做的工作的数据结构,或者说,是一个工作单位。Fiber 的架构还提供了一种跟踪、规划、暂停和销毁工作的便捷方式。**<br />当 React 元素第一次转换为 Fiber 节点时,React 在 createFiberFromTypeAndProps函数中使用元素中的数据来创建 Fiber。在随后的更新中,React 会再次利用 Fiber 节点,并使用来自相应 React 元素的数据更新必要的属性。如果不再从 `render` 方法返回相应的 React 元素,React 可能还需要根据 `key` 属性来移动或删除层级结构中的节点。<br />_查看 ChildReconciler函数以查看 React 为现有 Fiber 节点执行的所有活动和相应函数的列表。_<br />因为React为每个 React 元素创建一个 Fiber 节点,并且因为我们有一个这些元素组成的树,所以我们可以得到一个 Fiber 节点树。对于我们的示例应用程序,它看起来像这样:<br />![](https://cdn.nlark.com/yuque/0/2019/webp/128853/1564754956660-11f55b6e-a1fb-4199-8462-0be70e8f1be4.webp#align=left&display=inline&height=242&originHeight=242&originWidth=555&size=0&status=done&width=555)<br />所有 Fiber 节点都通过链表连接,具体是使用Fiber节点上的 `child`、`sibling` 和 `return` 属性。至于它为什么以这种方式工作,如果您还没有阅读过我的文章,更多详细信息请查看 [React 在 Fiber 中使用链表的方法和原因](https://medium.com/react-in-depth/the-how-and-why-on-reacts-usage-of-linked-list-in-fiber-67f1014d0eb7)。
### current 树及 workInProgress 树
在第一次渲染之后,React 最终得到一个 Fiber 树,它反映了用于渲染 UI 的应用程序的状态。这棵树通常被称为 **current 树(当前树)**。当 React 开始处理更新时,它会构建一个所谓的 **workInProgress 树(工作过程树)**,它反映了要刷新到屏幕的未来状态。<br />所有工作都在 `workInProgress` 树的 Fiber 节点上执行。当 React 遍历 `current` 树时,对于每个现有 Fiber 节点,React 会创建一个构成 `workInProgress` 树的备用节点,这一节点会使用 render 方法返回的 React 元素中的数据来创建。处理完更新并完成所有相关工作后,React 将准备好一个备用树以刷新到屏幕。一旦这个 `workInProgress` 树在屏幕上呈现,它就会变成 `current`树。<br />React 的核心原则之一是一致性。 React 总是一次性更新 DOM - 它不会显示部分中间结果。`workInProgress` 树充当用户不可见的「草稿」,这样 React 可以先处理所有组件,然后将其更改刷新到屏幕。<br />在源代码中,您将看到很多函数从 `current` 和 `workInProgress` 树中获取 Fiber 节点。这是一个这类函数的签名:
```js
function updateHostComponent(current, workInProgress, renderExpirationTime) {...}
```
每个Fiber节点持有`备用`域在另一个树的对应部分的引用。来自 `current` 树中的节点会指向 `workInProgress` 树中的节点,反之亦然。
### 副作用
我们可以将 React 中的一个组件视为一个使用 state 和 props 来计算 UI 表示的函数。其他所有活动,如改变 DOM 或调用生命周期方法,都应该被视为副作用,或者简单地说是一种效果。[文档中](https://reactjs.org/docs/hooks-overview.html)是这样描述的:<br />_您之前可能已经在 React 组件中执行数据提取,订阅或手动更改 DOM。我们将这些操作称为“副作用”(或简称为“效果”),因为它们会影响其他组件,并且在渲染过程中无法完成。_<br />您可以看到大多 state 和 props 更新都会导致副作用。既然使用副作用是工作(活动)的一种类型,Fiber 节点是一种方便的机制来跟踪除了更新以外的效果。每个 Fiber 节点都可以具有与之相关的副作用,它们可在 `effectTag` 字段中编码。<br />因此,Fiber 中的副作用基本上定义了处理更新后需要为实例完成的 [工作](https://github.com/facebook/react/blob/b87aabdfe1b7461e7331abb3601d9e6bb27544bc/packages/shared/ReactSideEffectTags.js)。对于宿主组件(DOM 元素),所谓的工作包括添加,更新或删除元素。对于类组件,React可能需要更新 refs 并调用 `componentDidMount` 和 `componentDidUpdate` 生命周期方法。对于其他类型的 Fiber ,还有相对应的其他副作用。
### 副作用列表
React 处理更新的素对非常迅速,为了达到这种水平的性能,它采用了一些有趣的技术。**其中之一是构建具有副作用的 Fiber 节点的线性列表,从而能够快速遍历。**遍历线性列表比树快得多,并且没有必要在没有副作用的节点上花费时间。<br />此列表的目标是标记具有 DOM 更新或其他相关副作用的节点。此列表是 `finishedWork` 树的子集,并使用 `nextEffect` 属性而不是 `current` 和 `workInProgress` 树中使用的 `child` 属性进行链接。<br />Dan Abramov 为副作用列表提供了一个类比。他喜欢将它想象成一棵圣诞树,「圣诞灯」将所有有效节点捆绑在一起。为了使这个可视化,让我们想象如下的 Fiber 节点树,其中标亮的节点有一些要做的工作。例如,我们的更新导致 `c2` 被插入到 DOM 中,`d2` 和 `c1` 被用于更改属性,而 `b2`被用于触发生命周期方法。副作用列表会将它们链接在一起,以便 React 稍后可以跳过其他节点:<br />![](https://cdn.nlark.com/yuque/0/2019/webp/128853/1564754956620-8176f526-da4f-4099-b339-07e22b914820.webp#align=left&display=inline&height=357&originHeight=357&originWidth=475&size=0&status=done&width=475)<br />可以看到具有副作用的节点是如何链接在一起的。当遍历节点时,React 使用 `firstEffect` 指针来确定列表的开始位置。所以上面的图表可以表示为这样的线性列表:<br />![](https://cdn.nlark.com/yuque/0/2019/webp/128853/1564754956572-87cc0b5c-77cb-4be0-915f-628f38e3f1c8.webp#align=left&display=inline&height=103&originHeight=103&originWidth=712&size=0&status=done&width=712)<br />如您所见,React 按照从子到父的顺序应用副作用。
<a name="FEgIh"></a>
### Fiber 树的根节点
每个 React 应用程序都有一个或多个充当容器的 DOM 元素。在我们的例子中,它是带有 ID 为 `container` 的 `div` 元素。React 为每个容器创建一个 Fiber 根对象。您可以使用对 DOM 元素的引用来访问它:
```jsx
const fiberRoot = query('#container')._reactRootContainer._internalRoot
```
这个 Fiber 根是React保存对 Fiber 树的引用的地方,它存储在 Fiber 根对象的 `current` 属性中:
```jsx
const hostRootFiberNode = fiberRoot.current
```
Fiber 树以一个特殊类型的 Fiber 节点 `HostRoot` 开始。它在内部创建的,并充当最顶层组件的父级。`HostRoot` 节点可通过 `stateNode` 属性返回到 `FiberRoot`:
```jsx
fiberRoot.current.stateNode === fiberRoot; // true
```
你可以通过 Fiber 根访问最顶层的 `HostRoot` 节点来探索 Fiber 树,或者可以从组件实例中获取单独的 Fiber 节点,如下所示:
```jsx
compInstance._reactInternalFiber
```
### Fiber 节点结构
现在让我们看一下为 `ClickCounter` 组件创建的 Fiber 节点的结构
```js
{
stateNode: new ClickCounter,
type: ClickCounter,
alternate: null,
key: null,
updateQueue: null,
memoizedState: {count: 0},
pendingProps: {},
memoizedProps: {},
tag: 1,
effectTag: 0,
nextEffect: null
}
```
以及 `span` DOM 元素:
```js
{
stateNode: new ClickCounter,
type: ClickCounter,
alternate: null,
key: null,
updateQueue: null,
memoizedState: {count: 0},
pendingProps: {},
memoizedProps: {},
tag: 1,
effectTag: 0,
nextEffect: null
}
```
Fiber 节点上有很多字段。我在前面的部分中描述了字段 `alternate`、 `effectTag` 和 `nextEffect` 的用途。现在让我们看看为什么需要其他字段。
#### stateNode
保存组件的类实例、DOM 节点或与 Fiber 节点关联的其他 React 元素类型的引用。总的来说,我们可以认为该属性用于保持与一个 Fiber 节点相关联的局部状态。
#### type
定义此 Fiber 节点的函数或类。对于类组件,它指向构造函数,对于 DOM 元素,它指定 HTML 标记。我经常使用这个字段来理解 Fiber 节点与哪个元素相关。
#### tag
定义 [Fiber 的类型](https://github.com/facebook/react/blob/769b1f270e1251d9dbdce0fcbd9e92e502d059b8/packages/shared/ReactWorkTags.js)。它在协调算法中用于确定需要完成的工作。如前所述,工作取决于React元素的类型。函数createFiberFromTypeAndProps将 React 元素映射到相应的 Fiber 节点类型。在我们的应用程序中,`ClickCounter` 组件的属性 `tag` 是 1,表示是 `ClassComponent`(类组件),而 `span` 元素的属性 `tag` 是 5,表示是 `HostComponent`(宿主组件)。
#### updateQueue
状态更新、回调和 DOM 更新的队列。
#### memoizedState
用于创建输出的 Fiber 状态。处理更新时,它会反映当前在屏幕上呈现的状态。
#### memoizedProps
在前一个渲染中用于创建输出的 Fiber 的 props。
#### pendingProps
已从 React 元素中的新数据更新并且需要应用于子组件或 DOM 元素的 props。
#### key
唯一标识符,当具有一组子元素的时候,可帮助 React 确定哪些项发生了更改、添加或删除。它与 [此处](https://reactjs.org/docs/lists-and-keys.html) 描述的 React 的「列表和键」功能有关。我在上面的解释中省略了一堆字段。特别是,我跳过了指针 `child`、`sibling` 和 `return`,它们构成了我在 [上一篇文章](https://medium.com/react-in-depth/the-how-and-why-on-reacts-usage-of-linked-list-in-fiber-67f1014d0eb7) 中描述的树数据结构。以及专属于 Scheduler 的 expirationTime,childExpirationTime 和 mode 等字段类别。
## 通用算法
React 在两个主要阶段执行工作:`render` 和 `commit`。<br />在第一个 `render` 阶段,React 通过 `setUpdate` 或 `React.render` 计划性的更新组件,并确定需要在 UI 中更新的内容。如果是初始渲染,React 会为 `render` 方法返回的每个元素创建一个新的 Fiber 节点。在后续更新中,现有 React 元素的 Fiber 节点将被重复使用和更新。**这一阶段是为了得到标记了副作用的 Fiber 节点树**。副作用描述了在下一个`commit`阶段需要完成的工作。在当前阶段,React持有标记了副作用的 Fiber 树并将其应用于实例。它遍历副作用列表、执行 DOM 更新和用户可见的其他更改。<br />**我们需要重点理解的是,第一个 `render` 阶段的工作是可以异步执行的。**React 可以根据可用时间片来处理一个或多个 Fiber 节点,然后停下来暂存已完成的工作,并转而去处理某些事件,接着它再从它停止的地方继续执行。但有时候,它可能需要丢弃完成的工作并再次从顶部开始。由于在此阶段执行的工作不会导致任何用户可见的更改(如 DOM 更新),因此暂停行为才有了意义。**与之相反的是,后续 `commit` 阶段始终是同步的。**这是因为在此阶段执行的工作会导致用户可见的变化,例如 DOM 更新。这就是为什么 React 需要在一次单一过程中完成这些更新。<br />React 要做的一种工作就是调用生命周期方法。一些方法是在 `render` 阶段调用的,而另一些方法则是在 `commit` 阶段调用。这是在第一个 `render` 阶段调用的生命周期列表:
- [UNSAFE_]componentWillMount(弃用)<br />
- [UNSAFE_]componentWillReceiveProps(弃用)<br />
- getDerivedStateFromProps<br />
- shouldComponentUpdate<br />
- [UNSAFE_]componentWillUpdate(弃用)<br />
- render<br />
正如你所看到的,从版本 16.3 开始,在 `render` 阶段执行的一些保留的生命周期方法被标记为 `UNSAFE`,它们现在在文档中被称为遗留生命周期。它们将在未来的 16.x 发布版本中弃用,而没有 `UNSAFE` 前缀的方法将在 17.0 中移除。您可以在 [此处](https://reactjs.org/blog/2018/03/27/update-on-async-rendering.html) 详细了解这些更改以及建议的迁移路径。<br />那么这么做的目的是什么呢?<br />好吧,我们刚刚了解到,因为 `render` 阶段不会产生像 DOM 更新这样的副作用,所以 React 可以异步处理组件的异步更新(甚至可能在多个线程中执行)。但是,标有 UNSAFE 的生命周期经常被误解和滥用。开发人员倾向于将带有副作用的代码放在这些方法中,这可能会导致新的异步渲染方法出现问题。虽然只有没有 UNSAFE 前缀的对应方法将被删除,但它们仍可能在即将出现的并发模式(您可以选择退出)中引起问题。<br />接下来罗列的生命周期方法是在第二个 `commit` 阶段执行的:
- getSnapshotBeforeUpdate<br />
- componentDidMount<br />
- componentDidUpdate<br />
- componentWillUnmount<br />
因为这些方法都在同步的 `commit` 阶段执行,他们可能会包含副作用,并对 DOM 进行一些操作。<br />至此,我们已经有了充分的背景知识,下面我们可以看下用来遍历树和执行一些工作的通用算法。
### Render 阶段
协调算法始终使用 renderRoot 函数从最顶层的 `HostRoot` 节点开始。不过,React 会略过已经处理过的 Fiber 节点,直到找到未完成工作的节点。例如,如果在组件树中的深层组件中调用 `setState` 方法,则 React 将从顶部开始,但会快速跳过各个父项,直到它到达调用了 `setState`方法的组件。
### 工作循环的主要步骤
所有的 Fiber 节点都会在工作循环中进行处理。如下是该循环的同步部分的实现:
```js
function workLoop(isYieldy) {
if (!isYieldy) {
while (nextUnitOfWork !== null) {
nextUnitOfWork = performUnitOfWork(nextUnitOfWork);
}
} else {...}
}
```
在上面的代码中,`nextUnitOfWork` 持有 `workInProgress` 树中的 Fiber 节点的引用,这个树有一些工作要做。当 React 遍历 Fiber 树时,它会使用这个变量来知晓是否有任何其他 Fiber 节点具有未完成的工作。处理过当前 Fiber 后,变量将持有树中下一个 Fiber 节点的引用或 null。在这种情况下,React 退出工作循环并准备好提交更改。<br />遍历树、初始化或完成工作主要用到 4 个函数:
- performUnitOfWork
- beginWork
- completeUnitOfWork
- completeWork
为了演示他们的使用方法,我们可以看看如下展示的遍历 Fiber 树的动画。我已经在演示中使用了这些函数的简化实现。每个函数都需要对一个 Fiber 节点进行处理,当 React 从树上下来时,您可以看到当前活动的 Fiber 节点发生了变化。从视频中我们可以清楚地看到算法如何从一个分支转到另一个分支。它首先完成子节点的工作,然后才转移到父节点进行处理。<br />![](https://user-gold-cdn.xitu.io/2018/12/3/1677442376e53cda?imageslim#align=left&display=inline&height=420&originHeight=420&originWidth=780&status=uploading&width=780)<br />_注意,垂直方向的连线表示同层关系,而折线连接表示父子关系,例如,`b1` 没有子节点,而 `b2`有一个子节点 `c1`。_<br />在这个 [视频](https://vimeo.com/302222454) 中我们可以暂停播放并检查当前节点和函数的状态。从概念上讲,你可以将「开始」视为「进入」一个组件,并将「完成」视为「离开」它。在解释这些函数的作用时,您也可以在 [这里](https://stackblitz.com/edit/js-ntqfil%3Ffile=index.js) 使用示例和实现。<br />我们首先开始研究 `performUnitOfWork` 和 `beginWork` 这两个函数:
```js
function performUnitOfWork(workInProgress) {
let next = beginWork(workInProgress);
if (next === null) {
next = completeUnitOfWork(workInProgress);
}
return next;
}
function beginWork(workInProgress) {
console.log('work performed for ' + workInProgress.name);
return workInProgress.child;
}
```
函数 `performUnitOfWork` 从 `workInProgress` 树接收一个 Fiber 节点,并通过调用 `beginWork`函数启动工作。这个函数将启动所有 Fiber 执行工作所需要的活动。出于演示的目的,我们只 log 出 Fiber 节点的名称来表示工作已经完成。**函数 `beginWork` 始终返回指向要在循环中处理的下一个子节点的指针或 null。**<br />如果有下一个子节点,它将被赋值给 `workLoop` 函数中的变量 `nextUnitOfWork`。但是,如果没有子节点,React 知道它到达了分支的末尾,因此它可以完成当前节点。**一旦节点完成,它将需要为同层的其他节点执行工作,并在完成后回溯到父节点。**这是 `completeUnitOfWork` 函数执行的代码:
```js
function completeUnitOfWork(workInProgress) {
while (true) {
let returnFiber = workInProgress.return;
let siblingFiber = workInProgress.sibling;
nextUnitOfWork = completeWork(workInProgress);
if (siblingFiber !== null) {
// If there is a sibling, return it
// to perform work for this sibling
return siblingFiber;
} else if (returnFiber !== null) {
// If there's no more work in this returnFiber,
// continue the loop to complete the parent.
workInProgress = returnFiber;
continue;
} else {
// We've reached the root.
return null;
}
}
}
function completeWork(workInProgress) {
console.log('work completed for ' + workInProgress.name);
return null;
}
```
你可以看到函数的核心就是一个大的 `while` 的循环。当 `workInProgress` 节点没有子节点时,React 会进入此函数。完成当前 Fiber 节点的工作后,它就会检查是否有同层节点。如果找的到,React 退出该函数并返回指向该同层节点的指针。它将被赋值给 `nextUnitOfWork` 变量,React将从这个节点开始执行分支的工作。我们需要着重理解的是,在当前节点上,React 只完成了前面的同层节点的工作。它尚未完成父节点的工作。**只有在完成以子节点开始的所有分支后,才能完成父节点和回溯的工作。**<br />从实现中可以看出,`performUnitOfWork` 和 `completeUnitOfWork` 主要用于迭代目的,而主要活动则在 `beginWork` 和 `completeWork` 函数中进行。在后续的系列文章中,我们将了解随着 React 进入 `beginWork` 和 `completeWork` 函数,`ClickCounter` 组件和 `span` 节点会发生什么。
### `commit` 阶段
这一阶段从函数 completeRoot 开始。在这个阶段,React 更新 DOM 并调用变更生命周期之前及之后方法的地方。<br />当 React 进入这个阶段时,它有 2 棵树和副作用列表。第一个树表示当前在屏幕上渲染的状态,然后在 `render` 阶段会构建一个备用树。它在源代码中称为 `finishedWork` 或 `workInProgress`,表示需要映射到屏幕上的状态。此备用树会用类似的方法通过 `child` 和 `sibling` 指针链接到 `current` 树。<br />然后,有一个副作用列表 -- 它是 `finishedWork` 树的节点子集,通过 `nextEffect` 指针进行链接。需要记住的是,副作用列表是运行 `render` 阶段的_结果_。渲染的重点就是确定需要插入、更新或删除的节点,以及哪些组件需要调用其生命周期方法。这就是副作用列表告诉我们的内容,**它页正是在 `commit` 阶段迭代的节点集合。**<br />_出于调试目的,可以通过 Fiber 根的属性 `current`访问 `current` 树。可以通过 `current` 树中 `HostFiber` 节点的 `alternate` 属性访问 `finishedWork` 树。_<br />在 `commit` 阶段运行的主要函数是 commitRoot 。它执行如下下操作:
- 在标记为 `Snapshot` 副作用的节点上调用 `getSnapshotBeforeUpdate` 生命周期<br />
- 在标记为 `Deletion` 副作用的节点上调用 `componentWillUnmount` 生命周期<br />
- 执行所有 DOM 插入、更新、删除操作<br />
- 将 `finishedWork` 树设置为 current<br />
- 在标记为 `Placement` 副作用的节点上调用 `componentDidMount` 生命周期<br />
- 在标记为 `Update` 副作用的节点上调用 `componentDidUpdate` 生命周期<br />
在调用变更前方法 `getSnapshotBeforeUpdate` 之后,React 会在树中提交所有副作用,这会通过两波操作来完成。第一波执行所有 DOM(宿主)插入、更新、删除和 ref 卸载。然后 React 将 `finishedWork` 树赋值给 `FiberRoot`,将 `workInProgress` 树标记为 `current` 树。这是在提交阶段的第一波之后、第二波之前完成的,因此在 `componentWillUnmount` 中前一个树仍然是 current,在 `componentDidMount/Update` 期间已完成工作是 current。在第二波,React 调用所有其他生命周期方法和引用回调。这些方法单独传递执行,从而保证整个树中的所有放置、更新和删除能够被触发执行。<br />以下是运行上述步骤的函数的要点:
```jsx
function commitRoot(root, finishedWork) {
commitBeforeMutationLifecycles()
commitAllHostEffects();
root.current = finishedWork;
commitAllLifeCycles();
}
```
这些子函数中都实现了一个循环,该循环遍历副作用列表并检查副作用的类型。当它找到与函数目的相关的副作用时,就会执行。
### 更新前的生命周期方法
例如,这是在副作用树上遍历并检查节点是否具有 `Snapshot` 副作用的代码:
```js
function commitBeforeMutationLifecycles() {
while (nextEffect !== null) {
const effectTag = nextEffect.effectTag;
if (effectTag & Snapshot) {
const current = nextEffect.alternate;
commitBeforeMutationLifeCycles(current, nextEffect);
}
nextEffect = nextEffect.nextEffect;
}
}
```
对于一个类组件,这一副作用意味着会调用 `getSnapshotBeforeUpdate` 生命周期方法。
### DOM 更新
commitAllHostEffects是 React 执行 DOM 更新的函数。该函数基本上定义了节点需要完成的操作类型,并执行这些操作:
```jsx
function commitAllHostEffects() {
switch (primaryEffectTag) {
case Placement: {
commitPlacement(nextEffect);
...
}
case PlacementAndUpdate: {
commitPlacement(nextEffect);
commitWork(current, nextEffect);
...
}
case Update: {
commitWork(current, nextEffect);
...
}
case Deletion: {
commitDeletion(nextEffect);
...
}
}
}
```
有趣的是,React 调用 `componentWillUnmount` 方法作为 `commitDeletion` 函数中删除过程的一部分。
### 更新后的生命周期方法
commitAllLifecycles 是 React 调用所有剩余生命周期方法的函数。在 React 的当前实现中,唯一会调用的变更方法就是 `componentDidUpdate`。<br />我们终于讲完了。让我知道您对该文章的看法或在评论中提出问题。我还有关于调度程序、子调和过程以及如何构建副作用列表的文章来对这些内容提供深入的解释。我还计划制作一个视频,在这里我将展示如何使用本文作为基础来调试应用程序。
---
## 公众号
想要实时关注笔者最新的文章和最新的文档更新请关注公众号**程序员面试官**,后续的文章会优先在公众号更新.
**简历模板:** 关注公众号回复「模板」获取
**《前端面试手册》:** 配套于本指南的突击手册,关注公众号回复「fed」获取
![2019-08-12-03-18-41]( https://xiaomuzhu-image.oss-cn-beijing.aliyuncs.com/d846f65d5025c4b6c4619662a0669503.png)