React函数式组件渲染、useEffect顺序总结
参考资料:
深入React的生命周期(上):出生阶段(Mount)
深入React的生命周期(下):更新(Update)
精读《useEffect 完全指南》
React组件重新渲染理解 & 优化大全
React渲染顺序及useEffect执行顺序探究(含并发模式)
组件状态
同样还是可以把组件的状态分为mount、update和unmount。
- mount:组件首次出现在页面中。React会通过最后组件的函数返回值,确定它有哪些子组件,依次mount和render子组件。一个component只是定义了,但没有最后被返回,就不会被挂载和渲染。
- update:组件的重新渲染,重新渲染的条件见这里。
- unmount:如果组件从页面消失,就会被unmount。常见于条件渲染、或者子组件未使用key标明时的位置更改。
React16的组件渲染过程
即使用ReactDOM.createRoot(DOM).render(<App />)渲染组件时。
单一组件渲染过程
在组件的不同阶段,调用顺序如下
mount
- 函数体:此时
useState等hooks取初始值,如果用callback初始化,则会调用初始化函数 - effect函数:会调用一遍所有
useEffect注册的函数,调用顺序就是useEffect在函数体里出现的顺序
update
函数体:正常调用,取最新state和ref的值
clean函数:如果依赖项
A=[…]发生改变,则会调用,但若有其它依赖项B也变了,却没列进依赖项里,这些未注册依赖项会使用最后一次A=[…]发生改变时的B的值。因为这是clean函数最新的定义。样例可见React函数式组件渲染顺序探究(Demo),组件依赖了name和state,但只注册了state这一个依赖项。
effect函数:如果依赖项
A=[…]发生改变,则会调用,但若有其它依赖项B也变了,却没列进依赖项里,这些未注册依赖项会使用最后一次A=[…]发生改变时的B的值。因为这是effect函数最新的定义。
unmount
clean函数:会调用一遍所有
useEffect返回的clean函数,调用顺序也是注册顺序。同样,也取的是它最新的定义。假设有两个effect,都有未注册依赖项
B。但它们一个依赖项为A=[…],另一个为[]。如果最开始
B=1,而A变的时候B=2,最后unmount的时候,前者的B=2,后者的B=1,因为后者的clean函数并未更新。
树型组件调用顺序
mount
如果有一个这样的component:
<A>
<A1>
<A1_1/>
<A1_2/>
</A1>
<A2>
<A2_1/>
<A2_2/>
</A2>
</A>
- 函数体:调用顺序是先序遍历的DFS,即
[A, A1, A1_1, A1_2, A2, A2_1, A2_2] - effect:类似于二叉树的后序遍历,先遍历孩子,再遍历根,即:
[A1_1, A1_2, A1, A2_1, A2_2, A2, A]。
update
如果上述的component变成了如下,A重新渲染。
<A>
<A2>
<A2_1/>
<A2_2/>
</A2>
<A1>
<A1_1/>
<A1_2/>
</A1>
</A>
- 如果A1和A2没有设置key,React会当作需要unmount旧的A1、A2,再mount新的A1、A2。
- 否则,React只会重新渲染A1、A2。
假设这里A1设置了key,而A2没有:
- 函数体:按当前组件内容的先序DFS:
[A1, A1_1, A1_2, A2, A2_1, A2_2]。 - clean:
- 先执行unmount的组件的clean,执行顺序是先序DFS,即
[A2, A2_1, A2_2] - 再执行update组件的clean,执行顺序是后序DFS,即
[A1_1, A1_2, A1]
- 先执行unmount的组件的clean,执行顺序是先序DFS,即
- effect:按当前组件内容的后序DFS执行,即
[A2_1, A2_2, A2, A1_1, A1_2, A1]。
React18的更新
即使用StrictMode渲染组件时。
root.render(
<React.StrictMode>
<App />
</React.StrictMode>
);
与16最大的区别是:
(1)函数体都会被调用2遍
(2)新mount的组件都会再次调用一遍clean和effect
有点类似于, mount (React 18)约等于mount(React 16) + update(React 16)
所以:
第一次body即mount的body,第二次body是update的body
但也不完全相同,比如,如果使用callback来初始化state的值时,mount的时候调用的两遍函数体,都是会调用这个callback的,而不是一次调用一次不调用。
第一次effect是mount的effect
接下来的clean和effect是update时的clean+effect
mount
如果有一个这样的component:
<A>
<A1>
<A1_1/>
<A1_2/>
</A1>
<A2>
<A2_1/>
<A2_2/>
</A2>
</A>
- 函数体:调用顺序是先序遍历的DFS,即
[A, A1, A1_1, A1_2, A2, A2_1, A2_2]。由于会叫两遍,实际上是[A, A, A1, A1, A1_1, A1_1, A1_2, A1_2, A2, A2, A2_1, A2_1, A2_2, A2_2]- 有意思的是,不是先mount DFS一遍,update DFS一遍,是在DFS的过程中直接叫两遍函数体。
- effect:类似于二叉树的后序遍历,先遍历孩子,再遍历根,即:
[A1_1, A1_2, A1, A2_1, A2_2, A2, A]。 - clean:后序DFS
[A1_1, A1_2, A1, A2_1, A2_2, A2, A]。 - effect:后序DFS
[A1_1, A1_2, A1, A2_1, A2_2, A2, A]。
update
如果上述的component变成了如下,A重新渲染。
<A>
<A2>
<A2_1/>
<A2_2/>
</A2>
<A1>
<A1_1/>
<A1_2/>
</A1>
</A>
- 如果A1和A2没有设置key,React会当作需要unmount旧的A1、A2,再mount新的A1、A2。
- 否则,React只会重新渲染A1、A2。
假设这里A1设置了key,而A2没有:
- 函数体:按当前组件内容的先序DFS:
[A1, A1_1, A1_2, A2, A2_1, A2_2]。同样会调用2次。 - clean:
- 先执行unmount的组件的clean,执行顺序是先序DFS,即
[A2, A2_1, A2_2] - 再执行update组件的clean,执行顺序是后序DFS,即
[A1_1, A1_2, A1]
- 先执行unmount的组件的clean,执行顺序是先序DFS,即
- effect:按当前组件内容的后序DFS执行,即
[A2_1, A2_2, A2, A1_1, A1_2, A1]。 - clean:mount的组件会被update,所以会有第二轮clean,后序DFS,即
[A2_1, A2_2, A2] - effect:第二轮effect,即
[A2_1, A2_2, A2]
React函数式组件渲染、useEffect顺序总结的更多相关文章
- 如何对 React 函数式组件进行优化
文章首发个人博客 前言 目的 本文只介绍函数式组件特有的性能优化方式,类组件和函数式组件都有的不介绍,比如 key 的使用.另外本文不详细的介绍 API 的使用,后面也许会写,其实想用好 hooks ...
- React函数式组件和类组件[Dan]
一篇对Dan的 How Are Function Components Different from Classes? 一文的个人阅读总结,内容来自于此.强烈推荐阅读 Dan Abramov.的博客. ...
- React函数式组件使用Ref
目录: 简介 useRef forwardRef useImperativeHandle 回调Ref 简介 大家都知道React中的ref属性可以帮助我们获取子组件的实例或者Dom对象,进而对子组件进 ...
- React函数式组件的性能优化
优化思路 主要优化的方向有2个: 减少重新 render 的次数.因为在 React 里最重(花时间最长)的一块就是 reconction(简单的可以理解为 diff),如果不 render,就不会 ...
- React 函数式组件的 Ref 和子组件访问(useImperativeHandle)
引入:如何调用函数式组件内部的方法 对于 React 中需要强制修改子组件的情况,React 提供了 Refs 这种解决办法,使得我们可以操作底层 DOM 元素或者自定的 class 组件实例.除此之 ...
- React函数式组件值之useRef()和useImperativeHandle()
一.useRef useRef共有两种用法,获取子组件的实例(只有类组件可用),在函数组件中的一个全局变量,不会因为重复 render 重复申明, 类似于类组件的 this.xxx. 1. useRe ...
- react中类组件、函数组件、state、单层遍历、多层遍历、先遍历后渲染、if-else、三目运算符
1.回顾 module.exports = { entry: {}, output: {}, plugins: [], module: {}, resolve: {}, devServe: {} } ...
- 函数式组件中实现Antd打开Modal后其Input框自动聚焦(focus)到文字的最后
目前React使用函数式组件已经成为趋势, 如何把React函数式组件用好, 提高性能, 从而实现业务需求也成为了一种能力的体现......咳咳咳, 进入正题: 现实场景需求 我想实现这一个需求, 父 ...
- react 中的无状态函数式组件
无状态函数式组件,顾名思义,无状态,也就是你无法使用State,也无法使用组件的生命周期方法,这就决定了函数组件都是展示性组件,接收Props,渲染DOM,而不关注其他逻辑. 其实无状态函数式组件也是 ...
- 渲染函数render和函数式组件
vnode对象 vnode对象包括(vnode并不是vue实例,而是vue实例中渲染函数render执行后生成的结果) this.tag = tag // 当前节点标签名 this.data = da ...
随机推荐
- Maven的大概了解及总结setting和pom
前言:项目中经常要用到Maven,从来也没有配置过,直到当人问到Maven是干什么的,是怎么管理项目的?一头雾水,所以写了这篇博客,首先附上百度百科的词条: Maven项目对象模型(POM),可以通过 ...
- 一个WPF开发的、界面简洁漂亮的音频播放器
今天推荐一个界面简洁.美观的.支持国际化开源音频播放器. 项目简介 这是一个基于C# + WPF开发的,界面外观简洁大方,操作体验良好的音频播放器. 支持各种音频格式,包括:MP4.WMA.OGG.F ...
- 关于OA系统的取数依据,以及如何逆向查询数据错误的思路。
1.正文 00.起因 源于财务在群里发的问题.我估计对于很多同事,又会像往常一样充满疑问,尤其是对于oa的取数会有疑问,然后业务能力极差的那部分,又会想到这是运营的问题(话说关运营什么事?),这是项目 ...
- 【Python基础】推导式(列表、字典、集合)
推导式是一种简洁.高效的语法,用于从一个可迭代对象中生成新的可迭代(iterable)对象. 通常情况下,在以下情况可以考虑使用推导式: 只需要简单的表达式来计算新的可迭代对象的元素. 可迭代对象不是 ...
- OData WebAPI实践-OData与EDM
本文属于 OData 系列 引言 在 OData 中,EDM(Entity Data Model) 代表"实体数据模型",它是一种用于表示 Web API 中的结构化数据的格式.E ...
- mysql 5.7 json 类型 json 数组类型 普通字符串类型 10w数据 查询速度差异
json 非数组 建表语句ddl CREATE TABLE tb_json_test ( id INT NOT NULL AUTO_INCREMENT, user_no VARCHAR(100), u ...
- Django4全栈进阶之路5 Model模型
在 Django 中,模型(Model)是用于定义数据结构的组件,其作用如下: 定义数据结构:模型用于定义数据库中的表格和表格中的字段(列),其中每个模型类对应一个表格,模型中的每个字段对应表格中的一 ...
- 数据库优化案例—某市中心医院HIS系统
记得在自己学习数据库知识的时候特别喜欢看案例,因为优化的手段是容易掌握的,但是整体的优化思想是很难学会的.这也是为什么自己特别喜欢看案例,今天也开始分享自己做的优化案例. 最近一直很忙,博客产出也少的 ...
- Android Studio历史版本下载地址汇总
原文地址: Android Studio历史版本下载地址汇总 - Stars-One的杂货小窝 由于新公司不给自带电脑,然后给了台新的电脑,于是就是需要重新下载Android Studio 但众所周知 ...
- Google Chrome 超详细使用教程
由于微信不允许外部链接,你需要点击文章尾部左下角的 "阅读原文",才能访问文中的链接. 调查统计机构 NetMarketShare 发布最新的 7 月份报告,在全球浏览器市场,谷歌 ...