setState是同步的还是异步的

42 阅读3分钟

这是一个经典的 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 内部机制的理解深度。