React中setState是同步的还是异步的

 更新时间:2026年06月10日 08:53:28   作者:卷帘依旧  
这篇文章主要介绍了React中setState是同步的还是异步的,这是一个经典的 React 面试题,但答案并不是简单的“同步”或“异步”,需要分情况讨论,下面就来详细的介绍一下

这是一个经典的 React 面试题,但答案并不是简单的“同步”或“异步”,需要分情况讨论。

核心结论:在 React 能控制的事件中(如合成事件、生命周期),setState 是“异步”的;在 React 无法控制的地方(如原生事件、setTimeout、Promise 等),setState 是同步的。

这个“异步”是批量更新策略带来的效果,而不是真的异步代码。

1. 为什么会有“异步”的表现?

这是 React 为了性能优化而设计的批量更新机制。

  • 批量更新:如果你在一个事件处理函数中多次调用 setState,React 不会立即更新 this.state,而是将它们收集起来,只触发一次重新渲染。这避免了不必要的渲染,提升了性能。
  • 表现:因为更新被“推迟”了,在调用 setState 后立刻读取 this.state,你得到的还是旧值,所以感觉像是异步的。

2. 具体情况分析

情况一:React 可控制的范围(“异步”/批量更新)

场景: React 的合成事件(onClickonChange 等)、React 生命周期函数(componentDidMount 等)。

表现: 不会立即更新,会进行批量更新。

handleClick = () => {
  console.log(this.state.count); // 0
  this.setState({ count: this.state.count + 1 });
  console.log(this.state.count); // 还是 0(“异步”,还没更新)
  
  this.setState({ count: this.state.count + 1 });
  console.log(this.state.count); // 依然是 0
}
// 函数执行完后,React 会批量更新,count 只增加 1 次(因为两次 setState 读到的 state.count 都是旧值 0)

情况二:脱离 React 控制的范围(同步)

场景: 原生 DOM 事件监听、setTimeoutsetInterval、Promise.then 等。

表现: 立即更新,每一次 setState 都会触发一次重新渲染。

handleClick = () => {
  setTimeout(() => {
    console.log(this.state.count); // 0
    this.setState({ count: this.state.count + 1 });
    console.log(this.state.count); // 1(同步,立即更新了)
    
    this.setState({ count: this.state.count + 1 });
    console.log(this.state.count); // 2
  }, 0);
}
// 在 setTimeout 中,每次 setState 都会立即生效并渲染

原因: React 的批量更新是通过一个“开关”控制的。在 React 自身的事件处理函数开始时,这个开关是打开的,函数结束后统一更新并关闭。而 setTimeout 里的代码执行时,这个开关已经是关闭状态,所以每次 setState 都会立即生效。

3. 如何在“异步”状态下获取更新后的值?

React 提供了几种方式:

  1. setState 的回调函数:这是最直接的方式,回调会在更新完成后执行。

    this.setState({ count: this.state.count + 1 }, () => {
      console.log(this.state.count); // 这里拿到的是更新后的值
    });
    
  2. componentDidUpdate 生命周期:在组件更新后被调用,可以在里面对比新旧 props/state。

  3. 在 useEffect 中监听(函数组件):将 count 作为依赖项,当它变化时执行逻辑。

    useEffect(() => {
      console.log(count); // count 更新后会打印最新值
    }, [count]);
    

4. 总结对比

场景setState 行为是否立即更新 state是否多次渲染
React 合成事件 (onClick)异步(批量更新)否(一次)
React 生命周期 (componentDidMount)异步(批量更新)否(一次)
原生事件 (addEventListener)同步是(多次)
异步函数 (setTimeout, Promise)同步是(多次)

5. 加分点:React 18 中的变化

如果你对 React 18 比较熟悉,可以补充说明,这会是加分项。

React 18 开始,引入了新的 createRoot API,所有React.startTransition 或并发特性包裹下的更新,都会默认进行自动批处理。这意味着即使在 setTimeoutPromise 等异步回调中,setState 也是批量异步的。

// React 18 + createRoot
setTimeout(() => {
  setCount(c => c + 1);
  setFlag(f => !f);
  // React 18 也会将这两个更新合并为一次渲染
}, 1000);

小结:回答这个问题时,先抛出核心结论(分场景),再解释原因(批量更新机制),最后提一下 React 18 的变化。这样能清晰地展示你对 React 内部机制的理解深度。

到此这篇关于React中setState是同步的还是异步的的文章就介绍到这了,更多相关React中setState同步还是异步内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • React-Router6版本的更新引起的路由用法变化

    React-Router6版本的更新引起的路由用法变化

    本文主要介绍了React-Router6版本的更新引起的路由用法变化,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-01-01
  • 解决React报错Style prop value must be an object

    解决React报错Style prop value must be a

    这篇文章主要为大家介绍了React报错Style prop value must be an object解决,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-12-12
  • React + Threejs + Swiper 实现全景图效果的完整代码

    React + Threejs + Swiper 实现全景图效果的完整代码

    全景图效果非常漂亮给人带来极好的用户体验效果,那么基于前端开发如何实现这种效果呢,下面小编给大家带来了React + Threejs + Swiper 实现全景图效果,感兴趣的朋友一起看看吧
    2021-06-06
  • react中使用signal的项目实践

    react中使用signal的项目实践

    本文主要介绍了react中使用signal的项目实践,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2026-03-03
  • react 实现图片正在加载中 加载完成 加载失败三个阶段的原理解析

    react 实现图片正在加载中 加载完成 加载失败三个阶段的原理解析

    这篇文章主要介绍了react 实现图片正在加载中 加载完成 加载失败三个阶段的,通过使用loading的图片来占位,具体原理解析及实现代码跟随小编一起通过本文学习吧
    2021-05-05
  • React 中实现代码分割的方法与实践指南

    React 中实现代码分割的方法与实践指南

    这篇文章给大家介绍了React中实现代码分割的方法,包括路由级代码分割和组件级代码分割,并阐述了其原理和实际应用场景,本文结合实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧
    2026-01-01
  • 浅谈react-native热更新react-native-pushy集成遇到的问题

    浅谈react-native热更新react-native-pushy集成遇到的问题

    下面小编就为大家带来一篇浅谈react-native热更新react-native-pushy集成遇到的问题。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-09-09
  • React中getDefaultProps的使用小结

    React中getDefaultProps的使用小结

    React中的getDefaultProps功能允许开发者为类组件定义默认属性,提高组件的灵活性和容错性,本文介绍了getDefaultProps的作用、语法以及最佳实践,并探讨了其他替代方案,如函数组件中的默认参数、高阶组件和ContextAPI等,理解这些概念有助于提升代码的可维护性和用户体验
    2024-09-09
  • React+valtio响应式状态管理

    React+valtio响应式状态管理

    Valtio是一个很轻量级的响应式状态管理库,使用外部状态代理去驱动React视图来更新,本文主要介绍了React+valtio响应式状态管理,感兴趣的可以了解一下
    2023-12-12
  • react中useRef的应用使用详解

    react中useRef的应用使用详解

    这篇文章主要介绍了react中useRef的应用使用详解的相关资料,需要的朋友可以参考下
    2023-05-05

最新评论