setState 同步异步
在 React 中,setState
的执行时机依赖于它所在的执行环境。具体来说:
1)异步情况 :当 setState
在 React 的生命周期方法(如 componentDidMount
、componentDidUpdate
)或者合成事件(如捕获用户操作的事件处理函数)中被调用时,setState
是异步的,会自动进行批处理更新。这种异步性是为了提高性能,避免频繁渲染和批处理多个 setState
调用。
在 React 内部,调用 setState
后,新的状态并不会立即改变,而是被放入一个更新队列。这些更新会在某个时间点批量处理。为什么要这样做呢?这是为了优化性能和资源利用,防止频繁和冗余的组件重新渲染。
2)同步情况 :当 setState
在非 React 控制的环境中被调用,比如原生事件处理函数(如 addEventListener
注册的回调)或者定时器(如 setTimeout
、setInterval
)中被调用时,React 并不会对这些调用进行批处理,setState
是同步的。这是因为这些回调并不在 React 的合成事件系统或者生命周期管理之中。他们不受 React 的管理,React 无法确保可以安全地批处理这些更新。
3)实践中如何处理 setState
:
- 防止更新重复 :在合成事件和生命周期方法中,如果你需要基于前一个状态计算新的状态,应该使用
setState
的函数式调用形式:this.setState((prevState) => ({ ... }))
,确保状态计算的准确性。这也是因为解决可能批量更新导致的问题:无法获取状态更新中途的数据(因为批量更新只能控制到最新的状态)。这样的情况会每次更新后重新 render,否则都是在批量更新后 render 一次。 - 确保更新顺序 :有时我们需要保证多次
setState
调用的顺序,可以将多个状态更新合并到一次调用中,或者利用callback
参数来确保某个操作在状态更新后(组件重新渲染后)执行,一般用于获取最新值:this.setState(newState, callback)
。
4)react Fiber 的影响 : 在新的 React Fiber 架构中,异步渲染提供了更精细的控制,使得 setState
的行为更加高效。Fiber 允许中断渲染和继续执行,更好地调配资源,这也是为什么 setState
的更新有时表现为异步。
setState 批量更新
React 的 setState
批量更新是异步的。
为了性能原因,React 会将多个 setState
调用合并成一次批量更新。具体过程如下:
1)React 先将调用的每个 setState
所产生的更新对象存储在一个队列中。
2)在所有的同步代码执行完之后,React 调度机制会统一处理这些队列中的更新,将多次 setState 进行合并(将多次状态改变完成后,再统一对 state 进行改变,然后触发 render),批量处理。
3)React 根据更新对象对组件进行合并更新,只在最后一次调用时实际渲染更新后的 DOM。
需要注意:
-
setState 之所以设计成异步更新是为了保证性能,避免每次 setState 都引起组件的重新渲染。异步 setState ,合并多个状态更新统一处理,减少渲染次数,提高性能。
-
react 使用事务机制实现批量更新,React 的批处理既包括状态更新(state),也包括生成并应用这些更新的 DOM 操作。
-
setState 的第二个参数是一个回调函数,会在 setState 更新(因为批量更新,所以实际上是所有状态更新全部完成后才会执行这个回调函数)并重新渲染组件后被调用,所以这时我们可以获取最新值。
jsthis.setState({ counter: this.state.counter + 1 }, () => { console.log("State updated:", this.state.counter); });
-
强制同步更新,类组件可以通过
forceUpdate
强制同步更新,函数式组件使用useSyncExternalStore
强制同步更新。- 强制同步更新场景:如果组件的
render
方法是直接读取外部数据源时(而不是 render 中读取了 this.props, this.state 等,因为这样当你在组件或其任一父组件内调用setState
时,它就将自动重新渲染),则必须告诉 React 在该数据源更改时更新用户界面。
- 强制同步更新场景:如果组件的
-
react18 之前,合成事件和生命周期方法中 setState 是批量更新的(异步的);原生事件或者 setTimeout 中,setState 是同步的。react18之后,默认情况下都是批量更新。