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 求值时机。)

相关推荐
vim怎么退出18 小时前
Dive into React——Hooks 原理
react.js·源码阅读
光影少年21 小时前
react的useMemo 如何优化?
前端·react.js·掘金·金石计划
YFF菲菲兔21 小时前
React 核心流程总述
react.js
光影少年1 天前
react状态管理
前端·react.js·前端框架
珎珎啊1 天前
React 和 Vue 3的区别
前端·vue.js·react.js
Bigger1 天前
mini-cc 终端 UI:用 React 写 CLI 是什么体验
前端·react.js·ai编程
吹个口哨写代码1 天前
IIS 部署 Vue/React 单页应用 (SPA) 刷新页面 404/403.18 报错原因及终极解决方案
前端·vue.js·react.js
喵个咪1 天前
基于 Taro 的 Headless CMS 多端前端架构:技术解析与二次开发导引
前端·react.js·taro
假如让我当三天老蒯2 天前
React+TS 项目结构(自学项目用)
前端·react.js