傻子都能理解的 React Hook 闭包陷阱

为什么开发时总有一些状态没有得到更新,明明已经更新了状态?挠破头也想不明白。

这时你可能已经陷入了一个React新手朋友常入的坑点------闭包陷阱。

想要理解这一点,还是要从原理层下手。本文会将原理层的代码抽象成非常易懂的简化代码,让大家彻底明白React是怎么把你"耍"了的。

一. 先来发现Hook不能怎样使用

合法的本质:hook在每次组件更新时,必须保证相同的执行顺序。

一切不能保证相同执行顺序的逻辑都不能用。所以只能在组件顶层调用,下面的做法都是非法的:

1. 嵌套

js 复制代码
function MyComponent() {
  function handleClick() {
    const [count, setCount] = useState(0); // ❌ 错误!函数的执行不一定在时候,固然顺序不对。
    // ...
  }
  
  return <button onClick={handleClick}>Click me</button>;
}

2. 循环

js 复制代码
function MyComponent({ items }) {
  for (let i = 0; i < items.length; i++) {
    const [value, setValue] = useState(items[i]); // ❌ 错误!每次循环的次数、循环体的逻辑实现都未必保证每次组件函数执行,都完全一致。
    // ...
  }
  // ...
}

3. 条件语句

js 复制代码
function MyComponent({ shouldFetch }) {
  if (shouldFetch) {
    const [data, setData] = useState(null); // ❌ 错误!更不用说了,每次都可能执行或者不执行
    // ...
  }
  // ...
}

二. 为什么必须保证调用顺序完全一致?

结论是,React依赖Hook的调用顺序来关联渲染状态。比如:useState。

js 复制代码
function Counter({ isVisible }) {
  const [count, setCount] = useState(0);
  const [name, setName] = useState('Alice');
}

useState的本质是一个函数,每次调用useState都是创建不同的状态变量,而useState无论调用多少次始终都是同一个函数指针。这也就意味着useState不是"纯函数",每次执行都可能返回完全不同的值, 上面代码中第一次调用返回的是count, setCount,第二次调用返回的是name, setName为了hook的"非纯函数"的特性,React通过链表来维护了每一个Hook的状态

useState的逻辑可以被简化成下面的逻辑:

js 复制代码
let currentHook = 0; // 当前Hook的索引,每个state都有一个状态
const memoryCells = []; // 数组,用来存放每次调用useState后创建的值。在React中这里实际应该是链表

function useState(initialState) {
  const hookIndex = currentHook++;
  
  if (!memoryCells[hookIndex]) {
    memoryCells[hookIndex] = initialState;
  }
  
  function setState(state) {/.修改memoryCells[hookIndex]的值./}
  
  return [memoryCells[hookIndex], setState]
}

function renderComponent() {
  currentHook = 0; // 每次渲染重置索引
  // ...调用组件函数
  const [count, setCount] = useState(0); // 返回memoryCells[0]
  const [name, setName] = useState('Alice'); // 返回memoryCells[1]
}

看了上面的代码应该就明白为什么顺序这么重要了。因为每次执行组件函数,都会将currentHook重置为0,每遇到一个useState都会递增currentHook,从而拿取 memoryCellscurrentHook 的值。如果useState的执行顺序混乱,或者执行次数也每次不固定,那么递增的索引会使得每次useState拿到的值都可能是错的。

三. 那结合其他Hooks呢?

其实都差不多,都是同一个hook函数每次调用返回的不一样状态。在React中,这些Hook状态都是通过链表来维护的。但是,所有Hooks都共享一条链表。

比如下面这段代码:

js 复制代码
function App(){
    const [count, setCount] = useState(1)
    const [name, setName] = useState('FDirector')
    useEffect(()=>{
        // ...
    }, [])
    const text = useMemo(()=>{
        return 'ddd'
    }, [])
}

在React中会被建立出这样一条Hooks链表:

那链表上的每个节点的数据结构(每个hook的状态载体)是什么样的?

可以被简化成下面这样,实际React中的实现会非常复杂,我们只需要知道个大概即可:

ts 复制代码
type Hook = {
  memoizedState: any, // 存储当前的组件state,useState能用的上
  baseState: any, 
  baseUpdate: Update<any, any> | null, 
  queue: UpdateQueue<any, any> | null,
  next: Hook | null, // 下一个Hook节点
};

好了,现在让我们抛弃经验意识,从原理层上思考下面的这段代码。渲染完毕后触发click事件,看看结果如何?

js 复制代码
function App(){
    const [count, setCount] = useState(1);
    useEffect(()=>{
        setInterval(()=>{
            console.log(count)
        }, 1000)
    }, [])
    function click(){ setCount(2) }
}

过程如下,稍显墨迹,但很细节,若有错误感谢指点:

  1. useState创建hook节点,memoizedState储存为1,并且将1存储在 第一轮App函数作用域的count变量上。

  2. useEffect创建节点,上一个hook节点的next指向它。当前为初始化阶段,依赖数组为\[\],创建 setInterval 的回调函数,作用域链中包含第一轮App函数作用域 ,count读取的是第一轮App函数作用域中的count。

  3. setInterval回调触发,console第一轮App函数作用域中的count;

  4. 点击事件触发,setCount参数为2,使得useState hook的memoizedState = 2。App函数重新执行,进入第二轮App函数作用域

  5. useState读取hook节点的memoizedState,得到值为2,并将2存储在 第二轮App函数作用域 的count变量上。

  6. useEffect执行,前进hooks链表节点,但依赖数组为\[\],即无依赖变化,无需重新执行useEffect回调。故没有新的setInterval被创建,第一轮App函数作用域 中创建的 interval 也未得到销毁。

  7. setInterval回调触发,读取作用域链,为第一轮App函数作用域 ,得到的count故来自第一轮App函数作用域即依旧为1 !

相关推荐
kyriewen5 小时前
别再 console.log 了:5 个 Chrome DevTools 调试技巧,用过就回不去了
前端·javascript·面试
IT_陈寒6 小时前
Python搞不定字符串编码?这破玩意坑我两小时!
前端·人工智能·后端
DigitalOcean8 小时前
Laravel 开发者已在 DigitalOcean 上开通超过 10 万台服务器
前端·laravel
星始流年8 小时前
从 Tool 到 Skill——基于 LangChain 的服务端Skill实现
前端·langchain·agent
李惟8 小时前
开源本地通信库,纯客户端 RPC,像聊天一样通信
前端
YAwu118 小时前
深入解析 React 炫彩鼠标跟随标题组件:从坐标定位到动画性能
前端·react.js
GuWenyue8 小时前
排序效率低?5分钟吃透快速排序,性能飙升至O(nlogn)
前端·javascript·面试
OpenTiny社区8 小时前
🎨 看完 GenUI SDK 源码我悟了!
前端·vue.js·github
叁两8 小时前
前端转型AI Agent该如何学习?(前置篇)
前端·人工智能·node.js
何时梦醒8 小时前
深入理解递归与快速排序 —— 从基础入门到手写实现
前端·javascript