如何避免 React hooks 闭包陷阱?

sxkk20082年前知识分享189

什么是 React hooks 闭包陷阱?

在 react 中 提供了一些性能优化函数 react.memouseMemouseCallback

const cachedValue = useMemo((fn) => calculateValue, dependencies)

useMemo:memoized 值,只有依赖项变更的时候才会重新计算

const cachedFn = useCallback(fn, dependencies)

useCallback:memoized 函数,只有依赖项变更的时候才会重新更新

const MemoizedComponent = memo(SomeComponent, arePropsEqual?)

memo:缓存组件,当 props 没变化的时候,不会执行 render。arePropsEqual 是一个可选函数,可以自定义对比新旧的 props, 返回 true 就会缓存,返回 false,就不会缓存。

const arePropsEqual=(oldProps: Props, newProps: Props) => boolean

有时候我们使用了这些函数来优化性能,这些函数与外围的 state 形成闭包,导致缓存函数中获取到的 state 不是最新的值,这就是闭包陷阱。

实例演示

比如下面代码,在项目中有一个计时器组件,还有一个 Child 组件, 点击 Child 组件需要返回 App 组件中的最新 state 值;

import React, { useCallback, useEffect, useLayoutEffect } from 'react'
import 'antd/dist/antd.css'
import { Button, ButtonProps } from 'antd'

const Child = ({ onClick }: ButtonProps) => {
  console.log('render')
  return (
    <Button onClick={onClick} type="primary">
      Button
    Button>
  )
}

const App: React.FC = () => {
  const [count, setCount] = React.useState(0)

  useEffect(() => {
    const time = setInterval(() => {
      setCount((count) => count + 1)
    }, 1000)
    return () => {
      clearInterval(time)
    }
  }, [])

  const handleClick = () => {
    console.log(count)
  }

  return (
    <>
      <h2>{count}h2>
      <Child onClick={handleClick} />
    >
  )
}

export default App

这样没什么问题,但是每次渲染的时候 Child 组件都会执行 render

image.png

为了防止 App 组件在更新的时候,不重复渲染(render)子组件,我们使用 React.memo 包裹下 Child 组件, handleClick 也需要使用 useCallback 包裹,这样 Child 组件只会 render 一次。

const Child = React.memo(({ onClick }: ButtonProps) => {
  console.log('render')
  return (
    <Button onClick={onClick} type="primary">
      Button
    Button>
  )
})
const handleClick = useCallback(() => {
    console.log(count);
  }, []);

这样一来 useCallback 和 state 就形成了一个闭包,每次打印的 state 就是初始化的 state。

image.png

为了获得最新的 state 值,必须将 count 参数写进 useCallback 的第二个参数。

const handleClick = useCallback(() => {
  console.log(count)
}, [count])

但这样,又会导致 Child 组件更新。那么有什么好的解决办法呢?既能防止子组件的更新,又可以获取到最新的 state 值呢?

方法:

我们可以使用 useRef 来存一个函数,每次更新的时候设置 ref.current 的值,通过函数来获取最新的 state 值。

const App: React.FC = () => {
  const [count, setCount] = React.useState(0);
  const ref = React.useRef();

  useEffect(() => {
    const time = setInterval(() => {
      setCount((count) => count + 1);
    }, 1000);
    return () => {
      clearInterval(time);
    };
  }, []);

  const fn = () => {
    console.log(count);
  };

  ref.current = fn;

  const handleClick = useCallback(() => {
    ref.current();
  }, []);

  return (
    <>
      

{count}

); };

image.png

codesandbox 演示地址

小结

解决闭包陷阱的方法

  1. 当页面更新不频繁的时候,不使用 useMemouseCallback 缓存函数来优化页面;

  2. 将更新依赖的参数写进 useCallback 的第二个参数

  3. 使用 useRef 来存在一个函数,用一个函数实时获取最新的 state

以上就是本文全部内容,如果对你有帮助,可以随手点个赞,这对我真的很重要,希望这篇文章对大家有所帮助,也可以参考我往期的文章或者在评论区交流你的想法和心得,欢迎一起探索前端。

本文正在参加「金石计划 . 瓜分 6 万现金大奖」

相关文章

AI边缘计算盒子:连接未来的智能神器

AI边缘计算盒子:连接未来的智能神器

  智能科技正日益渗透进我们生活的方方面面。而AI边缘计算盒子作为一种创新设备,在连接未来的智能化世界中扮演着重要的角色。本文将为您介绍AI边缘计算盒子的原理和应用,带您领略...

人脑智能与人工智能的较量:未来谁将主宰?

人脑智能与人工智能的较量:未来谁将主宰?

  人类智慧的发展,源自人脑的高度进化。然而,随着科技的迅猛发展,人工智能也逐渐崭露头角。如今,人脑智能与人工智能正在展开一场精彩纷呈的较量,探讨着未来的主宰者。  人脑智能...

AI人工智能:未来智能科技的无限潜能与挑战

AI人工智能:未来智能科技的无限潜能与挑战

  随着科技的迅猛发展,AI人工智能成为了现代社会最为热门的话题之一。人们对AI的应用和未来潜力产生了极大的兴趣和期待。AI人工智能凭借其强大的计算能力和学习能力,正逐渐渗透到各行各业,为人类带来了巨...

AI农业:将人工智能技术应用于农业领域的创新之举

AI农业:将人工智能技术应用于农业领域的创新之举

  在当今社会,人工智能(AI)的发展正以惊人的速度改变着各行各业。其中,农业作为人类生存的基石,也不例外。AI农业,作为将人工智能技术应用于农业领域的创新之举,正在引起广泛...

人工智能的未来展望

人工智能的未来展望

  随着科技的不断发展,人工智能已经从科幻小说走到了现实生活中,它正悄然成为引领未来的一种核心技术。未来人工智能将给我们带来什么?如何开发和应对人工智能时代?本文将从理论和实...

AI检测:开启新媒体时代的智能呼唤

AI检测:开启新媒体时代的智能呼唤

  AI(人工智能)技术的不断发展和普及,已经渗透到各行各业中。其中,AI检测作为新兴应用领域之一,正在改变我们的生活方式和工作方式。借助AI检测技术,我们能够更加高效准确地...

发表评论    

◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。