【React】setState 的同步异步问题

setState 同步异步

在 React 中,setState 的执行时机依赖于它所在的执行环境。具体来说:

1)异步情况 :当 setState 在 React 的生命周期方法(如 componentDidMountcomponentDidUpdate)或者合成事件(如捕获用户操作的事件处理函数)中被调用时,setState 是异步的,会自动进行批处理更新。这种异步性是为了提高性能,避免频繁渲染和批处理多个 setState 调用。

在 React 内部,调用 setState 后,新的状态并不会立即改变,而是被放入一个更新队列。这些更新会在某个时间点批量处理。为什么要这样做呢?这是为了优化性能和资源利用,防止频繁和冗余的组件重新渲染。

2)同步情况 :当 setState 在非 React 控制的环境中被调用,比如原生事件处理函数(如 addEventListener 注册的回调)或者定时器(如 setTimeoutsetInterval)中被调用时,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。

需要注意:

  1. setState 之所以设计成异步更新是为了保证性能,避免每次 setState 都引起组件的重新渲染。异步 setState ,合并多个状态更新统一处理,减少渲染次数,提高性能。

  2. react 使用事务机制实现批量更新,React 的批处理既包括状态更新(state),也包括生成并应用这些更新的 DOM 操作。

  3. setState 的第二个参数是一个回调函数,会在 setState 更新(因为批量更新,所以实际上是所有状态更新全部完成后才会执行这个回调函数)并重新渲染组件后被调用,所以这时我们可以获取最新值。

    js 复制代码
    this.setState({ counter: this.state.counter + 1 }, () => {
        console.log("State updated:", this.state.counter);
    });
  4. 强制同步更新,类组件可以通过 forceUpdate 强制同步更新,函数式组件使用 useSyncExternalStore 强制同步更新。

    1. 强制同步更新场景:如果组件的 render 方法是直接读取外部数据源时(而不是 render 中读取了 this.props, this.state 等,因为这样当你在组件或其任一父组件内调用 setState 时,它就将自动重新渲染),则必须告诉 React 在该数据源更改时更新用户界面。
  5. react18 之前,合成事件和生命周期方法中 setState 是批量更新的(异步的);原生事件或者 setTimeout 中,setState 是同步的。react18之后,默认情况下都是批量更新。

相关推荐
学不会•34 分钟前
css数据不固定情况下,循环加不同背景颜色
前端·javascript·html
EasyNTS2 小时前
H.264/H.265播放器EasyPlayer.js视频流媒体播放器关于websocket1006的异常断连
javascript·h.265·h.264
活宝小娜3 小时前
vue不刷新浏览器更新页面的方法
前端·javascript·vue.js
程序视点3 小时前
【Vue3新工具】Pinia.js:提升开发效率,更轻量、更高效的状态管理方案!
前端·javascript·vue.js·typescript·vue·ecmascript
coldriversnow3 小时前
在Vue中,vue document.onkeydown 无效
前端·javascript·vue.js
我开心就好o3 小时前
uniapp点左上角返回键, 重复来回跳转的问题 解决方案
前端·javascript·uni-app
开心工作室_kaic4 小时前
ssm161基于web的资源共享平台的共享与开发+jsp(论文+源码)_kaic
java·开发语言·前端
刚刚好ā4 小时前
js作用域超全介绍--全局作用域、局部作用、块级作用域
前端·javascript·vue.js·vue
沉默璇年5 小时前
react中useMemo的使用场景
前端·react.js·前端框架
yqcoder5 小时前
reactflow 中 useNodesState 模块作用
开发语言·前端·javascript