WHAT - React 函数与 useMemo vs useCallback

embedded/2024/9/19 11:01:06/ 标签: react.js, 前端, 前端框架

目录

  • 一、介绍
    • `useMemo` 与 `useCallback` 的区别
    • 示例代码
      • `useMemo` 示例
      • `useCallback` 示例
    • 总结
  • 二、当一个函数被作为依赖项
    • `useMemo` 和 `useCallback` 的适用情况
    • 选择使用 `useCallback` 或 `useMemo`
    • 总结
    • 实际例子

一、介绍

在 React 中,useMemouseCallback 是两个用于性能优化的钩子,尽管它们有些相似,但各自有不同的用途和应用场景。理解它们的区别和适用情况可以帮助你更有效地管理组件的性能和状态。

useMemouseCallback 的区别

  1. useMemo:

    • 目的: 用于记忆化(缓存)计算结果。它接受一个计算函数和依赖项数组,只有当依赖项发生变化时,计算函数才会重新执行。
    • 用法: 当你需要避免在每次渲染时重复计算一个值时使用。比如,当计算一个复杂的值(如列表过滤、排序)时,可以使用 useMemo 来缓存计算结果,避免不必要的重新计算。
    • 语法:
      const memoizedValue = useMemo(() => computeExpensiveValue(a, b), [a, b]);
      
  2. useCallback:

    • 目的: 用于记忆化(缓存)函数。它接受一个函数和依赖项数组,只有当依赖项发生变化时,函数才会重新创建。
    • 用法: 当你需要避免在每次渲染时重新创建一个函数,尤其是当函数作为 props 传递给子组件时,使用 useCallback 可以确保子组件不因函数的重新创建而重新渲染。
    • 语法:
      const memoizedCallback = useCallback(() => {// Your callback logic
      }, [dependencies]);
      

示例代码

useMemo 示例

假设你有一个计算密集型的函数 computeExpensiveValue,它依赖于 ab

import React, { useMemo, useState } from 'react';const ExpensiveComponent = ({ a, b }) => {// 使用 useMemo 缓存计算结果const result = useMemo(() => computeExpensiveValue(a, b), [a, b]);return <div>{result}</div>;
};function computeExpensiveValue(a, b) {// 假设这个函数很复杂console.log('Computing expensive value...');return a + b;
}

在这个示例中,computeExpensiveValue 只会在 ab 改变时重新计算,否则会返回缓存的结果。

useCallback 示例

假设你有一个处理点击事件的函数 handleClick,它被传递给子组件 ButtonComponent

import React, { useCallback, useState } from 'react';const ParentComponent = () => {const [count, setCount] = useState(0);// 使用 useCallback 缓存函数const handleClick = useCallback(() => {setCount(count + 1);}, [count]);return <ButtonComponent onClick={handleClick} />;
};const ButtonComponent = ({ onClick }) => {console.log('ButtonComponent rendered');return <button onClick={onClick}>Click me</button>;
};

在这个示例中,handleClick 只会在 count 改变时重新创建,从而避免不必要的子组件重新渲染。

总结

  • useMemo: 用于缓存计算结果。适用于复杂计算、渲染时计算。
  • useCallback: 用于缓存函数。适用于避免函数重新创建,尤其是当函数作为 props 传递给子组件时。

选择使用 useMemo 还是 useCallback 取决于你的具体需求。如果你需要记忆化计算结果,使用 useMemo;如果你需要记忆化函数,使用 useCallback。通常,它们可以结合使用,以优化性能和避免不必要的渲染或计算。

二、当一个函数被作为依赖项

当你有一个函数被用作依赖项时,是否应该使用 useMemo 还是 useCallback 来封装它,取决于你的具体情况。下面是关于这两种钩子的详细指南,以帮助你做出决定:

useMemouseCallback 的适用情况

  1. useCallback:

    • 用途: 用于缓存函数的实例。它创建一个稳定的函数引用,只有当其依赖项发生变化时,才会重新创建该函数。

    • 使用场景:

      • 当你需要将函数传递给子组件作为 props,以避免子组件因为函数引用变化而重新渲染。
      • 当你需要确保函数在依赖项变化时保持稳定。
    • 示例:

      import React, { useCallback, useState } from 'react';const ParentComponent = () => {const [count, setCount] = useState(0);// 使用 useCallback 缓存函数const handleClick = useCallback(() => {setCount(count + 1);}, [count]); // 依赖项为 countreturn <ChildComponent onClick={handleClick} />;
      };const ChildComponent = ({ onClick }) => {console.log('ChildComponent rendered');return <button onClick={onClick}>Click me</button>;
      };
      
  2. useMemo:

    • 用途: 用于缓存计算结果。它记忆化计算结果,只有在依赖项发生变化时,才会重新计算。

    • 使用场景:

      • 当你有一个计算密集型的操作,且希望避免每次渲染都进行重复计算。
      • 当你需要缓存某个值而不是函数,以减少计算量。
    • 示例:

      import React, { useMemo, useState } from 'react';const ParentComponent = ({ items }) => {// 使用 useMemo 缓存计算结果const sortedItems = useMemo(() => {return [...items].sort(); // 假设这是一个耗时的排序操作}, [items]); // 依赖项为 itemsreturn <ChildComponent items={sortedItems} />;
      };const ChildComponent = ({ items }) => {return (<ul>{items.map((item, index) => (<li key={index}>{item}</li>))}</ul>);
      };
      

选择使用 useCallbackuseMemo

  • 函数作为依赖项: 如果你有一个函数作为依赖项,并且你希望这个函数的引用在组件的生命周期中保持稳定,使用 useCallback 是合适的。例如,如果你将这个函数作为 props 传递给子组件,useCallback 可以确保子组件不会因为函数引用的变化而重新渲染。

  • 计算结果作为依赖项: 如果你希望缓存一个值的计算结果,而这个值的计算依赖于某些状态或属性,使用 useMemo 是合适的。例如,如果你有一个计算密集型的值需要在依赖项变化时重新计算,但希望在依赖项没有变化时返回缓存的结果。

总结

  • 使用 useCallback: 当你需要稳定的函数引用,尤其是当这个函数作为依赖项或传递给子组件时。
  • 使用 useMemo: 当你需要缓存计算结果,以避免重复的计算操作。

实际例子

假设你有一个函数 fetchData 被用作依赖项,在 useEffect 中:

import React, { useEffect, useCallback } from 'react';const MyComponent = ({ fetchData }) => {// 假设你在 useEffect 中使用 fetchDatauseEffect(() => {fetchData(); // 只在 fetchData 变化时调用}, [fetchData]);// 使用 useCallback 缓存 fetchData(如果需要稳定性)const memoizedFetchData = useCallback(fetchData, [fetchData]);return <div>Content</div>;
};export default MyComponent;

在这个例子中,useCallback 用于确保 fetchData 函数的稳定引用。如果 fetchData 是从 props 中传递进来的,你可以使用 useCallback 来缓存它的引用。

总的来说,选择 useMemo 还是 useCallback 取决于你是否在意函数的稳定性还是计算结果的稳定性。


http://www.ppmy.cn/embedded/105180.html

相关文章

150. 逆波兰表达式求值【 力扣(LeetCode) 】

一、题目描述 给你一个字符串数组 tokens &#xff0c;表示一个根据 逆波兰表示法 表示的算术表达式。 请你计算该表达式。返回一个表示表达式值的整数。 注意&#xff1a; 有效的算符为 ‘’、‘-’、‘*’ 和 ‘/’ 。每个操作数&#xff08;运算对象&#xff09;都可以是…

Linux3-Linux用户和权限

简介&#xff1a;个人学习分享&#xff0c;如有错误&#xff0c;欢迎批评指正。 一、root用户&#xff08;超级管理员&#xff09; 无论是Windows、MacOS、Linux均采用多用户的管理模式进行权限管理。 在Linux系统中&#xff0c;拥有最大权限的账户名为&#xff1a;root&#…

设计模式结构型模式之代理模式

结构型模式之代理模式 一、概念和使用场景1、概念2、核心思想3、java实现代理模式的方式4、使用场景 二、示例讲解1. 静态代理2. 动态代理 三、总结1、使用规则2、代理模式的优点包括&#xff1a;3、代理模式的缺点包括&#xff1a; 一、概念和使用场景 1、概念 代理模式是一…

如果你还不能完全明白什么是冒烟测试和回归测试的本质,那你就白干了

前言 对于很多刚入行的测试人员来说&#xff0c;对于冒烟测试和回归测试可能还不是十分熟悉&#xff0c;博为峰就和大家分享一下什么是冒烟测试与回归测试&#xff1f;如何才能做好它们&#xff1f; 1.何为冒烟测试 冒烟测试是自由测试的一种。冒烟测试在测试中发现问题&…

猎板PCB正片与负片工艺:深入解析PCB制造中的光刻技术及其应用

猎板PCB是一家专注于高品质印刷电路板(PCB)制造的领先企业。我们致力于为客户提供从设计到生产的一站式服务&#xff0c;包括但不限于单层、双层、多层PCB板的制造&#xff0c;以及复杂的正负片叠加工艺。 正负片叠加工艺在PCB制造中并不常见&#xff0c;因为正片和负片在光刻…

HTTP、Session、Token及Cookie详解

HTTP&#xff08;超文本传输协议&#xff09; HTTP&#xff08;Hypertext Transfer Protocol&#xff09; 是用于在Web浏览器和Web服务器之间传输信息的协议。它是无状态的&#xff0c;这意味着每个请求都是独立的&#xff0c;服务器不会记住之前的请求状态。HTTP使用客户端-服…

Unity AudioSource.PlayClipAtPoint听不到声音的一种情况

AudioSource.PlayClipAtPoint是按距离来控制音量的&#xff0c;所以距离太远是听不到的。而这个距离是播放音乐自动生成的GameObject名称是One shot audio的位置&#xff08;也就是输入的位置&#xff09;到AudioListener的位置。 AudioListener的位置是由其挂在的GameObject …

C++类和对象2

一.类的默认成员函数 类的默认成员函数是用户自己没有显示写出&#xff0c;而编译器会自动生成的成员函数。当需要用到这些函数时&#xff0c;编译器就会自动生成并调用。但是当用户显示写出时&#xff0c;以后再调用就会调用用户写的&#xff0c;编译器不再自动生成。下面就是…

深度解析C++中函数重载与引用

&#x1f308;个人主页&#xff1a;Yui_ &#x1f308;Linux专栏&#xff1a;Linux &#x1f308;C语言笔记专栏&#xff1a;C语言笔记 &#x1f308;数据结构专栏&#xff1a;数据结构 &#x1f308;C专栏&#xff1a;C 文章目录 1. 函数重载1.1 函数重载概念1.2 C支持函数重载…

数据库之心:MySQL 探索(一)mysql的安装和基本介绍

欢迎来到我们的MySQL博客&#xff01;在这里&#xff0c;我们将深入探讨MySQL数据库系统的各个方面&#xff0c;包括基础知识、优化技巧、实践案例以及最新的行业趋势。 目录 前言 什么是数据库&#xff1f; 数据库产品 MySQL安装 解压 配置 添加环境变量 初始化MySQL …

css基础学习总结(一)

文章目录 一. 选择器1. 标签选择器2. 类选择器3. ID选择器4. 分组选择器5. 派生与子元素选择器6. 属性选择器7. 伪类选择器8. 伪对象选择器9. 选择器的优先级别css优先级 的 6大分类 一. 选择器 1. 标签选择器 选择器例子描述elementp/div/span选择所有含有指定标签的元素 &…

每周12600元奖金池,邀你与昇腾算力共舞,openMind开发者盛宴启幕!

小伙伴们&#xff0c;是否瞬间被这个标题唤醒了在OpenI启智社区“我为开源打榜狂”黄金时代的温馨记忆&#xff1f;打榜活动虽已谢幕&#xff0c;但大家相伴度过12期的那份激情与创新的共鸣&#xff0c;促使OpenI启智社区在国产算力崛起的浪潮中勇立潮头&#xff0c;推出了“芯…

集成电路学习:什么是ARM先进精简指令集计算机

ARM&#xff1a;先进精简指令集计算机 ARM先进精简指令集计算机&#xff08;Advanced RISC Machine&#xff0c;简称ARM&#xff09;是一种基于精简指令集计算机&#xff08;RISC&#xff09;原则的计算机处理器架构&#xff0c;由英国的ARM公司开发。这种架构以其低功耗和高性…

【GPT】Coze使用开放平台接口-【6】Dify 也来一遍

前面讲了 coze 的相关用法&#xff0c;这边想着用 Dify 也来一遍&#xff0c;刚开始的时候接触的是 Dify&#xff0c;后面才是 coze。Dify 和 coze 的侧重点不同&#xff0c;我个人是更倾向用 Dify 构建工作流就可以了&#xff0c;coze 还是相对全能。 本节用 Dify 也会创建插…

代码随想录 刷题记录-24 图论 (1)理论基础 、深搜与广搜

一、理论基础 参考&#xff1a; 图论理论基础 深度优先搜索理论基础 广度优先搜索理论基础 dfs dfs搜索可一个方向&#xff0c;并需要回溯&#xff0c;所以用递归的方式来实现是最方便的。 有递归的地方就有回溯&#xff0c;例如如下代码&#xff1a; void dfs(参数) {…

MySQL-CGI-响应报文

账号密码涉及到&#xff1a; 与数据库已记录下来的进行校验。 如果也是来一个新建一个连接&#xff0c;必然效率低。 可以做一个数据库连接池&#xff0c;预生成数据库连接。 如何生成数据库连接&#xff1f;下面的流程做很多次。 然后放在链表中&#xff0c;&#xff08;池&am…

macos MacPort 包管理工具安装和使用

在macos v10.15版本中, xz, python等软件无法使用brew安装, 原因是brew对于旧版本的macos不再支持, 但是我们可以使用另外一个macos下的包管理工具来安装brew无法安装的软件, macport 是一个和brew类似的macos下的一个非常优秀的软件包安装管理工具. MacPort安装前提条件 安…

科研绘图系列:R语言基因PPI互作网络图(PPI network plot )

介绍 基因的PPT互作网络图。 加载R包 导入所需要的R包,在导入前需要用户自己安装。 library("tidyverse") library("magrittr") library("here") library("janitor") library("ggpubr") library("ComplexHeatmap&…

【STM32】位带操作

一、位带操作 1.意义 回想以前写51代码 ​ P0 0x10; //将P0端口设置为0x10 P1_01; //将P1端口0号引脚设置为高电平 a P2_2; //获取P2端口2号引脚的电平 根据上述的方法&#xff0c;我们可以发现快速定位修改某个引脚的电平还有获取引脚的状态 2.原因 GPIO_SetBits、GPI…

Ubuntu 22.04 镜像自定义制作

Centos封装镜像> Linux Centos ISO镜像自定义制作 Ubuntu20.04封装镜像> Ubuntu 20.04 镜像自定义制作 1、下载镜像 系统环境&#xff1a;ubuntu 22.04 镜像地址&#xff1a;https://mirrors.aliyun.com/ubuntu-releases rootubuntu:~# apt install -y p7zip xorriso ro…