傻子都能理解的 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,从而拿取 memoryCells[currentHook] 的值。如果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 !

相关推荐
摘星编程20 分钟前
React Native鸿蒙版:Image图片占位符
react native·react.js·harmonyos
未来之窗软件服务25 分钟前
未来之窗昭和仙君(六十五)Vue与跨地区多部门开发—东方仙盟练气
前端·javascript·vue.js·仙盟创梦ide·东方仙盟·昭和仙君
嘿起屁儿整38 分钟前
面试点(网络层面)
前端·网络
VT.馒头1 小时前
【力扣】2721. 并行执行异步函数
前端·javascript·算法·leetcode·typescript
phltxy2 小时前
Vue 核心特性实战指南:指令、样式绑定、计算属性与侦听器
前端·javascript·vue.js
Byron07073 小时前
Vue 中使用 Tiptap 富文本编辑器的完整指南
前端·javascript·vue.js
css趣多多3 小时前
地图快速上手
前端
zhengfei6113 小时前
面向攻击性安全专业人员的一体化浏览器扩展程序[特殊字符]
前端·chrome·safari
码丁_1173 小时前
为什么前端需要做优化?
前端
Mr Xu_4 小时前
告别硬编码:前端项目中配置驱动的实战优化指南
前端·javascript·数据结构