0

0

什么是useCallback?记忆化的函数

星降

星降

发布时间:2025-08-22 13:23:01

|

609人浏览过

|

来源于php中文网

原创

useCallback用于记忆化函数,避免组件重新渲染时函数引用变化导致子组件不必要的重渲染。它接收函数和依赖数组,仅当依赖项变化时返回新函数实例,常与React.memo配合优化性能,防止闭包陷阱需正确设置依赖,但不应过度使用,因有额外开销,适用于函数作为props传递至优化子组件的场景。

什么是usecallback?记忆化的函数

useCallback
是React提供的一个Hook,它的核心作用是记忆化(memoize)一个函数。简单来说,就是当你把一个函数作为props传递给子组件,或者这个函数本身被其他Hook(比如
useEffect
useMemo
)依赖时,
useCallback
可以确保这个函数在父组件重新渲染时,不会被“无缘无故”地重新创建,从而避免了不必要的子组件重新渲染或Hook的重复执行。

解决方案

在React中,每次组件重新渲染时,组件内部定义的函数都会被重新创建。这意味着,即使函数体本身没有变化,它的内存地址(引用)也会改变。这在大多数情况下不是问题,但当你将这个函数作为props传递给一个使用了

React.memo
(或类组件的
PureComponent
)进行优化的子组件时,问题就来了。

React.memo
会对其props进行浅比较,如果它收到的props与上次渲染的props在引用上相同,它就会跳过本次渲染。然而,如果一个函数prop在每次父组件渲染时都被重新创建,那么即使这个函数的功能完全一样,
React.memo
也会认为这个prop“变了”,从而导致子组件不必要的重新渲染。这无疑会抵消掉
React.memo
带来的性能优化。

useCallback
正是为了解决这个问题而生。它接收两个参数:一个要记忆化的函数,以及一个依赖项数组。
useCallback
会返回这个函数的记忆化版本。只有当依赖项数组中的某个值发生变化时,
useCallback
才会返回一个新的函数实例;否则,它会返回上一次渲染时记忆化的那个函数实例。

import React, { useState, useCallback, memo } from 'react';

// 一个使用React.memo优化的子组件
const ChildComponent = memo(({ onClick }) => {
  console.log('ChildComponent rendered');
  return ;
});

function ParentComponent() {
  const [count, setCount] = useState(0);
  const [text, setText] = useState('');

  // 没有使用 useCallback 的函数,每次 ParentComponent 渲染都会重新创建
  // const handleClick = () => {
  //   setCount(count + 1);
  //   console.log('Button clicked, count:', count);
  // };

  // 使用 useCallback 记忆化的函数
  const handleClick = useCallback(() => {
    setCount(prevCount => prevCount + 1); // 推荐使用函数式更新,避免将 count 加入依赖
    console.log('Button clicked, count:', count); // 注意:这里的 count 是定义时的值,如果需要最新值,用 prevCount
  }, []); // 依赖数组为空,表示这个函数只在组件初次挂载时创建一次

  const handleInputChange = (e) => {
    setText(e.target.value);
  };

  return (
    

Parent Count: {count}

Input Text: {text}

); } export default ParentComponent;

在这个例子中,如果你不使用

useCallback
,每次你在输入框中输入内容(导致
ParentComponent
重新渲染),
ChildComponent
也会跟着重新渲染,尽管它的
onClick
逻辑上并没有变。但加上
useCallback
后,只要
handleClick
的依赖项(这里是空数组,意味着没有依赖)不变,
ChildComponent
就不会因为
onClick
引用的变化而重新渲染。

为什么在React中需要useCallback?

这是一个很常见的问题,尤其是在你开始关注应用性能的时候。我们知道JavaScript里的函数也是对象,它们有自己的内存地址。在React组件每次重新渲染时,组件内部定义的任何函数,都会被视为一个新的函数实例,即使它们的代码一模一样。这就像你每次去银行,银行都给你发一张新的会员卡,虽然卡上的信息没变,但那确实是张“新卡”。

对于那些没有经过优化的普通React组件来说,这通常不是问题,因为它们无论如何都会重新渲染。但当你的子组件使用了

React.memo
(或类组件的
PureComponent
)进行优化时,问题就浮现了。
React.memo
的工作原理是对props进行浅层比较。如果它发现传递给它的函数prop的引用变了,它就会认为这个prop“变了”,然后触发子组件重新渲染。即使这个函数的功能、它内部依赖的数据都没有实际变化,仅仅是引用变了,
React.memo
的优化效果就大打折扣了。

所以,

useCallback
的必要性,主要体现在它能帮助我们维护函数引用的稳定性。通过确保函数引用不变,我们可以有效地配合
React.memo
工具,避免不必要的子组件重新渲染,从而提升应用的整体性能。尤其是在处理大量列表渲染、或者有复杂交互逻辑的组件时,这种优化能带来明显的流畅度提升。它不是一个万能药,但确实能解决特定场景下的性能瓶颈。

使用useCallback常见的误区或挑战有哪些?

尽管

useCallback
听起来很美好,但在实际使用中,它也常常会带来一些困惑和“坑”。

Voicenotes
Voicenotes

Voicenotes是一款简单直观的多功能AI语音笔记工具

下载

一个最常见的挑战就是依赖数组(dependency array)的处理。如果你忘记在依赖数组中包含函数内部使用的所有外部变量,你就会遇到“闭包陷阱”或者说“陈旧闭包”(stale closures)的问题。这意味着你的函数会捕获到它定义时的那个旧的变量值,而不是最新的值。比如,如果

handleClick
依赖于
count
,但你的依赖数组是
[]
,那么
handleClick
里面的
count
永远是组件初次渲染时的那个值。解决办法通常是将所有外部依赖都列入数组,或者,如果可以,使用函数式更新(如
setCount(prevCount => prevCount + 1)
)来避免对状态变量的直接依赖。

另一个误区是过度使用

useCallback
。不是所有的函数都需要被记忆化。
useCallback
本身也是有开销的,它需要额外的内存来存储记忆化的函数,并且每次渲染时都需要进行依赖项的比较。如果一个函数只是在组件内部使用,或者它被传递给一个没有经过
React.memo
优化的子组件,那么使用
useCallback
可能带来的性能提升微乎其微,甚至可能因为其自身的开销而导致负优化。判断是否需要用
useCallback
,一个简单的原则是:这个函数是否作为props传递给了
React.memo
包裹的子组件?或者它是否是另一个Hook(如
useEffect
,
useMemo
)的依赖?如果都不是,那很可能就不需要。

再有,就是误以为

useCallback
能解决所有性能问题。它仅仅是解决了函数引用变化导致的重复渲染问题。如果你的组件因为其他原因(比如状态更新频繁、计算量大、DOM操作复杂)而性能不佳,
useCallback
可能帮不上忙。它只是React性能优化工具箱中的一个特定工具,需要结合具体场景来使用。有时候,性能问题可能出在组件结构本身,或者数据流设计上,这时候就不是一个
useCallback
能解决的了。

useCallback与useMemo、React.memo有什么关系?

这三个概念在React的性能优化领域里,就像是亲密的三兄弟,它们各自有分工,但又紧密协作。理解它们之间的关系,对于掌握React的性能优化至关重要。

首先,我们来说

React.memo
。它不是Hook,而是一个高阶组件(Higher-Order Component, HOC)。它的作用是包裹一个函数组件,然后对这个组件的props进行浅层比较。如果props在引用上没有变化,
React.memo
就会阻止这个组件重新渲染。它是性能优化的“守门员”,决定一个组件是否需要重新渲染。

接着是

useMemo
。这个Hook和
useCallback
是同胞兄弟,它们都用于记忆化(memoization)。但它们的记忆化对象不同:
useMemo
记忆化的是一个。它接收一个函数和一个依赖数组,在依赖数组中的任何值发生变化时,它会重新执行这个函数并返回一个新的计算结果;否则,它返回上一次记忆化的结果。比如,你可以用它来记忆化一个计算开销很大的复杂数据结构,或者一个需要保持引用稳定的对象。

最后是我们的主角

useCallback
。它记忆化的是一个函数。从某种意义上说,
useCallback(fn, deps)
可以被看作是
useMemo(() => fn, deps)
的一个语法糖。
useMemo
返回的是
fn
执行后的结果,而
useCallback
直接返回
fn
这个函数本身,但保证了它的引用稳定。

它们之间的关系是这样的:

  • useCallback
    useMemo
    :它们都是为了提供“记忆化”能力,减少不必要的计算或对象/函数创建。
    useCallback
    useMemo
    在函数记忆化场景下的特化版本。
  • useCallback
    React.memo
    :它们是最佳搭档。当
    React.memo
    包裹的子组件接收一个函数作为prop时,为了让
    React.memo
    的浅比较能够有效工作,这个函数就需要通过
    useCallback
    来记忆化,从而确保它的引用稳定。如果函数引用每次都变,
    React.memo
    就会失效。

所以,它们共同构成了一个性能优化的链条:

useCallback
确保传递给子组件的函数引用稳定 ->
React.memo
利用这个稳定的引用来判断是否需要重新渲染子组件 ->
useMemo
则可以记忆化其他复杂的值,进一步减少不必要的计算。它们一起,帮助我们构建更高效、更流畅的React应用。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

198

2023.11.20

treenode的用法
treenode的用法

​在计算机编程领域,TreeNode是一种常见的数据结构,通常用于构建树形结构。在不同的编程语言中,TreeNode可能有不同的实现方式和用法,通常用于表示树的节点信息。更多关于treenode相关问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

538

2023.12.01

C++ 高效算法与数据结构
C++ 高效算法与数据结构

本专题讲解 C++ 中常用算法与数据结构的实现与优化,涵盖排序算法(快速排序、归并排序)、查找算法、图算法、动态规划、贪心算法等,并结合实际案例分析如何选择最优算法来提高程序效率。通过深入理解数据结构(链表、树、堆、哈希表等),帮助开发者提升 在复杂应用中的算法设计与性能优化能力。

17

2025.12.22

深入理解算法:高效算法与数据结构专题
深入理解算法:高效算法与数据结构专题

本专题专注于算法与数据结构的核心概念,适合想深入理解并提升编程能力的开发者。专题内容包括常见数据结构的实现与应用,如数组、链表、栈、队列、哈希表、树、图等;以及高效的排序算法、搜索算法、动态规划等经典算法。通过详细的讲解与复杂度分析,帮助开发者不仅能熟练运用这些基础知识,还能在实际编程中优化性能,提高代码的执行效率。本专题适合准备面试的开发者,也适合希望提高算法思维的编程爱好者。

26

2026.01.06

go语言闭包相关教程大全
go语言闭包相关教程大全

本专题整合了go语言闭包相关数据,阅读专题下面的文章了解更多相关内容。

137

2025.07.29

DOM是什么意思
DOM是什么意思

dom的英文全称是documentobjectmodel,表示文件对象模型,是w3c组织推荐的处理可扩展置标语言的标准编程接口;dom是html文档的内存中对象表示,它提供了使用javascript与网页交互的方式。想了解更多的相关内容,可以阅读本专题下面的文章。

3307

2024.08.14

PHP 高并发与性能优化
PHP 高并发与性能优化

本专题聚焦 PHP 在高并发场景下的性能优化与系统调优,内容涵盖 Nginx 与 PHP-FPM 优化、Opcode 缓存、Redis/Memcached 应用、异步任务队列、数据库优化、代码性能分析与瓶颈排查。通过实战案例(如高并发接口优化、缓存系统设计、秒杀活动实现),帮助学习者掌握 构建高性能PHP后端系统的核心能力。

101

2025.10.16

PHP 数据库操作与性能优化
PHP 数据库操作与性能优化

本专题聚焦于PHP在数据库开发中的核心应用,详细讲解PDO与MySQLi的使用方法、预处理语句、事务控制与安全防注入策略。同时深入分析SQL查询优化、索引设计、慢查询排查等性能提升手段。通过实战案例帮助开发者构建高效、安全、可扩展的PHP数据库应用系统。

86

2025.11.13

俄罗斯Yandex引擎入口
俄罗斯Yandex引擎入口

2026年俄罗斯Yandex搜索引擎最新入口汇总,涵盖免登录、多语言支持、无广告视频播放及本地化服务等核心功能。阅读专题下面的文章了解更多详细内容。

134

2026.01.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 4.2万人学习

Pandas 教程
Pandas 教程

共15课时 | 1.0万人学习

ASP 教程
ASP 教程

共34课时 | 4.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号