React useState 的懒初始化机制

前言

在 React 中,useState 可以接收两种类型的参数:

  1. 普通初始值

    scss 复制代码
    const [state, setState] = useState(0);
  2. 懒初始化函数

    scss 复制代码
    const [state, setState] = useState(() => computeExpensiveValue());

懒初始化函数的核心目标是:避免在非首次渲染时执行昂贵的初始计算


1. 懒初始化函数的行为

用一个例子对比两种写法:

scss 复制代码
function expensiveInit() {
  console.log('计算初始值');
  return 42;
}

// 普通方式
const [count1, setCount1] = useState(expensiveInit());

// 懒初始化方式
const [count2, setCount2] = useState(() => expensiveInit());

输出顺序

  • 普通方式:计算初始值 每次渲染都会打印
  • 懒初始化方式:计算初始值 只在首次渲染打印一次

为什么会这样?我们需要看 React 的底层工作机制。


2. React 内部是如何处理的

React 使用 Fiber 架构 来管理函数组件的状态。每个组件对应一个 Fiber 节点,useState 通过 hook 链表存储状态信息。

2.1 useState 底层步骤

  1. 创建或获取 hook 对象

    每次调用 useState,React 会检查当前 Fiber 的 hook 链表是否已经存在对应的 hook:

    • 首次渲染:创建 hook 对象,存储状态和更新队列
    • 更新渲染:直接从 hook 链表读取上一次的状态
  2. 判断传入参数类型

    • 如果传入 函数,React 会 判断是否是第一次执行:

      ini 复制代码
      if (typeof initialState === 'function') {
        hook.memoizedState = initialState(); // 仅首次渲染执行
      } else {
        hook.memoizedState = initialState;   // 直接使用值
      }
    • 如果是普通值,直接赋值给 hook.memoizedState

  3. 更新机制

    setState 被调用时,React 会把新的状态加入更新队列,然后调度重新渲染:

    • 再次渲染时 不会再次执行懒初始化函数
    • React 会直接使用 hook 中存储的状态或更新队列计算的新状态

2.2 Fiber 链表与 hook 链表

每个函数组件的状态存储在一个 hook 链表 中:

rust 复制代码
FiberNode
 ├── memoizedState -> Hook (useState)
 ├── memoizedState -> Hook (useState)
 └── ...

每个 Hook 对象结构大概如下:

yaml 复制代码
{
  memoizedState: 初始状态或最新状态,
  queue: 状态更新队列,
  next: 下一个 Hook
}
  • 首次渲染

    • React 会遍历 hook 链表,创建每个 hook 对象
    • 如果 hook 接收的是函数,React 会立即执行它来获得初始状态
  • 更新渲染

    • React 直接读取 hook 链表里的状态
    • 传给 useState 的函数 不会再执行

3. 为什么使用懒初始化

假设我们有一个复杂计算:

bash 复制代码
function expensiveInit() {
  let sum = 0;
  for (let i = 0; i < 1e8; i++) sum += i;
  return sum;
}
  • 如果写成普通方式:

    scss 复制代码
    const [value, setValue] = useState(expensiveInit());

    每次渲染都会执行这个循环,导致性能严重下降。

  • 如果使用懒初始化:

    scss 复制代码
    const [value, setValue] = useState(() => expensiveInit());

    循环只在首次渲染执行一次,后续渲染直接复用状态。


4. 总结

所谓的懒初始化 是通过向 useState 传入函数,使得初始化逻辑只在首次渲染时由 React 调用,避免组件每次执行时都提前计算初始值,从而减少不必要的性能开销。(懒初始化优化的不是 React 的行为,而是 JS 求值时机。)

相关推荐
kyriewen16 小时前
我用 50 行代码重写了 React Router 核心,终于搞懂了前端路由原理
前端·javascript·react.js
ZhengEnCi1 天前
Q02-Vue-React-index.html完全指南
vue.js·react.js·html
weedsfly1 天前
JavaScript 事件流:彻底搞懂捕获、冒泡与事件委托
前端·javascript·react.js
光影少年3 天前
原生DOM操作在React 中的注意事项
前端·javascript·react.js
YAwu114 天前
深入解析 React 炫彩鼠标跟随标题组件:从坐标定位到动画性能
前端·react.js
Ruihong4 天前
🎉 VuReact 1.9.0 发布,支持 Vue 3.4 defineModel 编译到 React
vue.js·react.js·面试
spmcor4 天前
React 架构师之路:Next.js 全栈革命(第八篇)
前端·react.js
假如让我当三天老蒯4 天前
React基础、进阶(学习用)
前端·react.js·面试
spmcor4 天前
为什么页面越用越卡?——React组件内存泄漏的排查与修复
react.js
天蓝色的鱼鱼5 天前
React Router v8 来了:react-router-dom 没了,老项目该怎么迁移?
前端·react.js