setState是同步更新还是异步更新
先说结论
React18之前:使用了ReactDOM.render
,setState
在React
调度流程中是异步更新,在原生事件和setTimeout
中是同步更新。
React18:使用ReactDOM.createRoot
,默认都是批量更新,也就是异步更新。
在React18之前,
setState
在原生事件和定时器中是同步的,在合成事件和生命周期函数里面是异步的,原理在于合成事件和生命周期函数调用顺序在批处理和更新之前,导致在合成事件和生命周期函数里没法立刻拿到更新后的值,导致形成所谓的异步
setState为什么设计为异步
1. 提升性能
如果每次调用setState
都进行一次更新,意味着render
函数会被频繁调用,界面重新渲染,最好的方法就是获取到多个更新之后,批量进行更新
2. 保持state和props同步
如果同步更新了state
,但是还没有执行render
函数,那么state
和props
不能保持同步
react18之前为什么不确定是同步还是异步呢
在
React
中,如果是由React
引发的事件处理(比如通过onClick
引发的事件处理),调用setState
不会同步更新this.state
;如果是绕过React
通过addEventListener
直接添加的事件处理函数,还有通过setTimeout/setInterval
产生事件处理,调用setState
会同步更新this.state
。
为什么会出现以上有时同步,有时异步的现象呢
- 在
React
的setState
函数实现中,会根据一个变量isBatchingUpdates
判断是直接更新this.state
还是放到队列中回头再说, - 而
isBatchingUpdates
默认是false
,也就表示setState
会同步更新this.state
, - 但是,有一个函数
batchedUpdates
,这个函数会把isBatchingUpdates
修改为true
, - 而当
React
在调用事件处理函数之前就会调用这个batchedUpdates
,从而react
控制的事件处理过程setState
不会同步更新this.state
。 - 即:
setState
的“异步”并不是说内部由异步代码实现的,其实本身执行的过程和代码都是同步的,只是合成事件和钩子函数的调用顺序在更新之前,导致在合成事件和钩子函数中没法立马拿到更新后的值,形式了所谓的“异步”, - 当然可以通过第二个参数
setState(partialState, callback)
中的callback
拿到更新后的结果。
react18之后setState有哪些改动
- react18之后,
setState
都会表现为异步(即批处理)。 - 批处理:是指
React
将多个状态更新分组到单个重新渲染中以获得更好的性能 - 如果同一点击事件中有两个状态更新,
React
总是将它们批处理为一次重新渲染。