React state 更新时机

设置 state 只会为下一次渲染变更 state 的值
一个 state 变量的值永远不会在一次渲染的内部发生变化
React 会使 state 的值始终"固定"在一次渲染的各个事件处理函数内部
React 会等到事件处理函数中的所有代码都运行完毕再处理 state 更新

  1. 在一个函数中,多次设置state, 最终结果由最后一次决定
js 复制代码
export default function Counter() {
 const [number, setNumber] = useState(0);
 return (
   <>
     <h1>{number}</h1>
     <button onClick={() => {
     // 第一次执行时,number为0
       setNumber(number + 1); // 相当于setNumber(0 + 1),React 准备在下一次渲染时将 number 更改为 1,React 将 "替换为 1" 添加到其队列中。
       setNumber(number + 1); // 相当于setNumber(0 + 1),React 准备在下一次渲染时将 number 更改为 1,React 将 "替换为 1" 添加到其队列中。
       setNumber(number + 1); // 相当于setNumber(0 + 1),React 准备在下一次渲染时将 number 更改为 1,React 将 "替换为 1" 添加到其队列中。
       // 执行结果 number = 1
     }}>+3</button>
   </>
 )
}
  1. 即使事件处理函数的代码是异步的,它获取到的state也是与本次最初执行时的值保持一致的,state在一次执行时不会改变。
js 复制代码
export default function Counter() {
  const [number, setNumber] = useState(0);
  return (
    <>
      <h1>{number}</h1>
      <button onClick={() => {
        setNumber(number + 5);
        setTimeout(() => {
          alert(number); // 0 在onClick执行时,number的值就被固定了。state 在"获取 UI 的快照"时就被"固定"了
        }, 3000);
      }}>+5</button>
    </>
  )
}

如何在重新渲染前,多次更新同一个state

setNumber(n => n + 1)

通过传入一个更新函数 n=>n+1来更新state的值

js 复制代码
export default function Counter() {
  const [number, setNumber] = useState(0);
  return (
    <>
      <h1>{number}</h1>
      <button onClick={() => {
        setNumber(n => n + 1); // n => n + 1 是一个函数。React 将它加入队列。
        setNumber(n => n + 1); // n => n + 1 是一个函数。React 将它加入队列。
        setNumber(n => n + 1); // n => n + 1 是一个函数。React 将它加入队列。
      }}>+3</button>
    </>
  )
}

当在下次渲染期间调用 useState 时,React 会遍历队列。 之前的 state 的值是 0,所以这就是 React 作为参数 n 传递给第一个更新函数的值。然后 React 会获取上一个更新函数的返回值,并将其作为 n 传递给下一个更新函数,以此类推:

更新队列 n 返回值
n => n + 1 0 0 + 1 = 1
n => n + 1 1 1 + 1 = 2
n => n + 1 2 2 + 1 = 3

React 会保存 3 为最终结果并从 useState 中返回。

相关推荐
m0_471199632 小时前
【场景】前端怎么解决离线收银、数据同步异常等场景问题
前端·javascript
栀秋6662 小时前
“无重复字符的最长子串”:从O(n²)哈希优化到滑动窗口封神,再到DP降维打击!
前端·javascript·算法
xhxxx2 小时前
不用 Set,只用两个布尔值:如何用标志位将矩阵置零的空间复杂度压到 O(1)
javascript·算法·面试
鹿鹿鹿鹿isNotDefined2 小时前
Antd5.x 在 Next.js14.x 项目中,初次渲染样式丢失
前端·react.js·next.js
有意义2 小时前
斐波那契数列:从递归到优化的完整指南
javascript·算法·面试
Mr.Jessy3 小时前
JavaScript高级:深入对象与内置构造函数
开发语言·前端·javascript·ecmascript
温宇飞3 小时前
深入理解 JavaScript 模块系统:CJS 与 ESM 的实现原理
javascript
幸运小圣3 小时前
深入理解ref、reactive【Vue3工程级指南】
前端·javascript·vue.js
用户47949283569153 小时前
面试官最爱挖的坑:用户 Token 到底该存哪?
前端·javascript·面试
Heo3 小时前
Vue3.4中diff算法核心梳理
前端·javascript·面试