【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之后,默认情况下都是批量更新。

相关推荐
Elena_Lucky_baby11 分钟前
实现路由懒加载的方式有哪些?
前端·javascript·vue.js
Domain-zhuo11 分钟前
如何利用webpack来优化前端性能?
前端·webpack·前端框架·node.js·ecmascript
理想不理想v16 分钟前
webpack如何自定义插件?示例
前端·webpack·node.js
小华同学ai33 分钟前
ShowDoc:Star12.3k,福利项目,个人小团队的在线文档“简单、易用、轻量化”还专门针对API文档、技术文档做了优化
前端·程序员·github
一雨方知深秋34 分钟前
智慧商城:封装getters实现动态统计 + 全选反选功能
开发语言·javascript·vue2·foreach·find·every
海威的技术博客36 分钟前
关于JS中的this指向问题
开发语言·javascript·ecmascript
王解1 小时前
Vue CLI 脚手架创建项目流程详解 (2)
前端·javascript·vue.js
刘大浪1 小时前
vue.js滑动到顶便锁定位置
前端·javascript·vue.js
小金刚®1 小时前
构建简洁之美:我的第一个前端页面
前端
ordinary902 小时前
指令-v-for的key
前端·javascript·vue.js