引言
作为 react 框架内置的 hooks 之一,我们知道这个 hook 可以实现缓存,进而提升页面性能。那么这个 hook 究竟可以在什么场景下使用呢?本文针对这个问题进行了探讨。
用法
1 | import { useCallback } from "react"; |
可以看到,useCallback 的入参有两个:函数和依赖。返回的是一个函数定义。
useCallback 可以缓存函数定义,只有当依赖中的数据更新后,才会重新创建函数,否则,使用的函数是上一次渲染时存储的函数。
理解 useCallback
- 普通的函数定义
通过点击按钮让数字增加的时候,由于 cnt 的 state 更新,导致整个组件重新渲染,函数 increment 重新创建。
- 普通函数包裹了 useCallback
increment 函数被 useCallback 包裹,当点击按钮时,cnt 更新,组件重新渲染。但是 cnt 加到 1 后就无法继续增长。
这是因为 useCallback 缓存了 increment 函数,每次点击按钮的时候,使用的是上一次缓存的函数,这个时候 cnt 的值都是 0,所以后面点击按钮时, cnt 无法继续往上加。
如果想让 cnt 继续累加,可以在 useCallback 的第二个参数上绑定依赖:[cnt]
。
但是这个做法并没有体现 useCallback 的缓存优势,更糟糕的是,这种错误使用 useCallback,需要额外产生监视 cnt 变化的性能消耗。
但是好在我们观察到了使用 useCallback 和不使用的区别。这是理解 useCallback 的第一步。
- useCallback 缓存的是什么
缓存的是具有相同内存地址的函数定义。每次重新创建一个函数,该变量都会指向不同的内存地址,useCallback 缓存的上上一个创建函数时的内存指针。
1 | "string" === "string" // true |
- 为什么要缓存函数定义?
如果函数作为 react 的 hooks 上的依赖,或者子组件的参数等场景,如果不使用 useCallback 进行缓存,每次函数创建都会返回一个新的函数定义(有点像创建一个变量,然后赋值空对象,这些对象执行的内存都不同,所以===时不相等),这样就会导致 hooks 内部重新执行或者子组件重新渲染。
useCallback 使用场景
- 减少组件重新渲染:子组件传入一个函数,且子组件被 React.memo 缓存了。
shippingForm 是一个人为增加了性能的组件,每次 count 增加的时候,该组件都要重新渲染,每次渲染性能都消耗很大,所以可以看到,点击+的时候,页面会出现很卡的效果。
但是当我们勾选(或取消)dark mode,发现这个页面切换是非流畅。这是因为 ShippingForm 并没有重新渲染。为什么呢?
每次切换白天/黑夜背景的时候,theme 的状态更新导致 productPage 重新渲染,由于 useCallback 的使用,handleSubmit 函数没有重新创建,所以传递给子组件 ShippingForm 的 props 没有改变,由于 React.memo 的使用,ShippingForm 没有重新渲染。
1 | <aside> |
- 减少函数重新创建
state 状态变化,导致页面重新渲染,为了减少函数创建,可以使用 useCallback 进行缓存
只看下面代码中的 useEffect 和 addGuessedLetter 函数,每次触发按键事件,guessedLetters 的值会更新,使得页面渲染,如果不使用 useCallback,addGuessedLetter 函数会在每次渲染的时候重新创建。
我们并不希望他重新创建,如果这个函数内部没有 setState,我们可以放到组件外面解决这个问题,但是这个函数有。为了减少函数创建,可以给这个函数包裹 useCallback,并且不要添加任何依赖。这样可以保证 addGuessedLetter 函数只被创建一次。尽管这个函数只被创建了一次,但是每次 setGuessedLetters 都会用到上一个 guessedLetters 的值,这是因为setGuessedLetters((curLetters) => [...curLetters, letter]);
总结
关于 useCallback 何时使用问题,其实主要把握一个点就行:useCallback 的功能就是缓存函数定义,如果你不希望函数由于重新创建产生新的定义,而导致多余渲染或者不必要重复执行,那就可以用 useCallback 进行缓存。