第五节:React Hooks进阶篇-如何用useMemo/useCallback优化性能

  • 反模式:滥用导致的内存开销
  • React 19编译器自动Memoization原理

React Hooks 性能优化进阶:从手动到自动 Memoization

(基于 React 18 及以下版本,结合 React 19 新特性分析)


一、useMemo/useCallback 的正确使用场景与优化策略

1. useMemo:缓存高开销计算结果

核心作用:避免每次渲染重复执行复杂计算(如数据过滤、数学运算)。
正确用法

const filteredList = useMemo(() => bigDataList.filter(item => item.category === activeCategory), [bigDataList, activeCategory] // 仅依赖变化时重新计算
);

适用场景
• 数据量大的列表过滤/排序
• 复杂对象/数组的派生状态(如用户权限树、图表数据预处理)
• 需稳定引用的对象(避免子组件因引用变化重新渲染)

2. useCallback:稳定函数引用

核心作用:避免函数因引用变化导致子组件无效渲染。
正确用法

const handleSubmit = useCallback((values) => api.postForm(values), [] // 空依赖:函数逻辑不依赖外部变量
);

适用场景
• 事件处理函数传递给 React.memo 优化的子组件
• 依赖闭包值的异步操作(如定时器、防抖函数)


二、反模式:滥用导致的性能陷阱

1. 过度缓存导致内存开销

问题:对简单计算或频繁变化的值使用 useMemo,反而增加内存和比较成本
示例

// ❌ 错误:简单计算无需缓存
const total = useMemo(() => a + b, [a, b]); 
2. 依赖项管理不当

缺失依赖:导致闭包中引用过期值

const [count, setCount] = useState(0);
const increment = useCallback(() => {setCount(count + 1); // ❌ 依赖缺失,始终基于初始count
}, []); 

冗余依赖

const fetchData = useCallback(() => {getData(userId); 
}, [userId, getData]); // ❌ getData若为稳定引用(如来自useCallback),则无需重复依赖
3. 忽略组件拆分优化

问题:依赖 useMemo 缓存大型组件渲染结果,而非拆分细粒度组件
优化方案

// ✅ 拆分子组件并用 React.memo 优化
const ExpensiveSection = memo(({ data }) => <div>{data}</div>);

三、React 19 编译器自动 Memoization 原理与影响

1. 自动优化的核心机制

智能依赖追踪:编译器静态分析组件代码,自动识别变量间的依赖关系
函数稳定性保证:即使父组件重新渲染,若函数逻辑未变化,编译器自动保持引用稳定
计算缓存:自动对高开销操作(如数组遍历、复杂运算)实施类似 useMemo 的优化

2. 开发者行为变化

代码简化:不再需要手动添加 useMemo/useCallback

// React 19 ✅ 自动优化
const filteredList = bigData.filter(item => item.category === activeCategory);

例外场景
• 第三方库依赖严格引用相等性(如某些动画库)
• 超高频更新场景(如实时游戏引擎)需手动干预

3. 性能优化优先级调整

优先策略

  1. 保持代码简洁,依赖编译器自动优化
  2. 使用性能分析工具定位真实瓶颈(如 React DevTools)
  3. 仅在必要时手动添加 Memoization

四、新旧版本性能优化对比

优化维度React 18(手动)React 19(自动)
代码复杂度高(需显式声明依赖和缓存)低(编译器自动处理)
内存占用可能因过度缓存增加按需优化,减少冗余缓存
维护成本高(需持续监控依赖关系)低(聚焦业务逻辑)
适用场景所有场景除极端性能敏感场景外全覆盖

五、最佳实践总结

  1. React 18 及以下版本
    • 对高频计算/函数传递场景精准使用 useMemo/useCallback
    • 通过 memo + useCallback 避免深层子组件无效渲染
  2. React 19 及以上版本
    • 优先编写直观代码,信任编译器优化能力
    • 升级后逐步移除冗余 Memoization 代码
  3. 通用原则
    • 避免过早优化,先用工具验证性能瓶颈
    • 保持组件细粒度化,减少单组件渲染压力

通过合理运用手动优化与编译器自动能力,可显著提升应用性能与代码可维护性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/76487.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

windows server C# IIS部署

1、添加IIS功能 windows server 2012、windows server 2016、windows server 2019 说明&#xff1a;自带的是.net 4.5 不需要安装.net 3.5 尽量使用 windows server 2019、2016高版本&#xff0c;低版本会出现需要打补丁的问题 2、打开IIS 3、打开iis应用池 .net 4.5 4、添…

Elasticsearch的Java客户端库QueryBuilders查询方法大全

matchAllQuery 使用方法&#xff1a;创建一个查询&#xff0c;匹配所有文档。 示例&#xff1a;QueryBuilders.matchAllQuery() 注意事项&#xff1a;这种查询不加任何条件&#xff0c;会返回索引中的所有文档&#xff0c;可能会影响性能&#xff0c;特别是文档数量很多时。 ma…

C#进阶学习(六)单向链表和双向链表,循环链表(下)循环链表

目录 &#x1f4ca; 链表三剑客&#xff1a;特性全景对比表 一、循环链表节点类 二、循环链表的整体设计框架 三、循环列表中的重要方法&#xff1a; &#xff08;1&#xff09;头插法&#xff0c;在头结点前面插入新的节点 &#xff08;2&#xff09;尾插法实现插入元素…

交换网络基础

学习目标 掌握交换机的基本工作原理 掌握交换机的基本配置 交换机的基本工作原理 交换机是局域网&#xff08;LAN&#xff09;中实现数据高效转发的核心设备&#xff0c;工作在 数据链路层&#xff08;OSI 模型第二层&#xff09;&#xff0c;其基本工作原理可概括为 “学习…

科学研究:怎么做

科研&#xff08;科学研究&#xff09;​​ 是指通过系统化的方法&#xff0c;探索自然、社会或人文领域的未知问题&#xff0c;以发现新知识、验证理论或解决实际问题的活动。它的核心是​​基于证据的探索与创新​​&#xff0c;旨在推动人类认知和技术的进步。 科研的核心要…

算法题(128):费解的开关

审题&#xff1a; 本题需要我们将多组测试用例中拉灯数小于等于6的最小拉灯数输出&#xff0c;若拉灯数最小值仍大于6&#xff0c;则输出-1 思路&#xff1a; 方法一&#xff1a;二进制枚举 首先我们先分析一下基本特性&#xff1a; 1.所有的灯不可能重复拉&#xff1a;若拉的数…

MFC文件-屏幕录像

下载本文件 本文件将获取屏幕图像数据的所有代码整合到两个文件中&#xff08;ScreenRecorder.h和ScreenRecorder.cpp&#xff09;&#xff0c;使获取屏幕图像数据变得简单。输出IYUV视频流。还可以获取系统播放的声音&#xff0c;输出PCM音频流。由于使用了MFC类&#xff0c;本…

0801ajax_mock-网络ajax请求1-react-仿低代码平台项目

0 vite配置proxy代理 vite.config.ts代码如下图所示&#xff1a; import { defineConfig } from "vite"; import react from "vitejs/plugin-react";// https://vite.dev/config/ export default defineConfig({plugins: [react()],server: {proxy: {&qu…

JVM笔记【一】java和Tomcat类加载机制

JVM笔记一java和Tomcat类加载机制 java和Tomcat类加载机制 Java类加载 * loadClass加载步骤类加载机制类加载器初始化过程双亲委派机制全盘负责委托机制类关系图自定义类加载器打破双亲委派机制 Tomcat类加载器 * 为了解决以上问题&#xff0c;tomcat是如何实现类加载机制的…

IP编址(来自YESLAB新网工的笔记)

上层协议类型 概念&#xff1a;通常指的是位于网络层&#xff08;如 IP 层&#xff09;以上的协议类型&#xff0c;这些协议在数据传输时需要由网络层&#xff08;或更低层&#xff09;协议承载。以 IP 协议为例&#xff0c;IP 报文头部中的 协议字段&#xff08;Protocol Fie…

SpringBoot学习(过滤器Filter。拦截器Interceptor。全局异常捕获处理器GlobalExceptionHandler)(详细使用教程)

目录 一、过滤器Filter。 1.1定义与规范。 1.2工作原理与范围。 1.3使用场景。 1.4 SpringBoot实现过滤器。&#xff08;Filter配置2种方式&#xff09; <1>注解配置(WebFilter、Order、ServletComponentScan)。 创建过滤器类。 启用 Servlet 组件扫描。 <2>配置类…

c++题目_P1443 马的遍历

P1443 马的遍历 # P1443 马的遍历 ## 题目描述 有一个 $n \times m$ 的棋盘&#xff0c;在某个点 $(x, y)$ 上有一个马&#xff0c;要求你计算出马到达棋盘上任意一个点最少要走几步。 ## 输入格式 输入只有一行四个整数&#xff0c;分别为 $n, m, x, y$。 ## 输出格式 …

清华《数据挖掘算法与应用》K-means聚类算法

使用k均值聚类算法对表4.1中的数据进行聚类。代码参考P281。 创建一个名为 testSet.txt 的文本文件&#xff0c;将以下内容复制粘贴进去保存即可&#xff1a; 0 0 1 2 3 1 8 8 9 10 10 7 表4.1 # -*- coding: utf-8 -*- """ Created on Thu Apr 17 16:59:58 …

HarmonyOS-ArkUI V2工具类:AppStorageV2:应用全局UI状态存储

AppStorageV2是一个能够跨界面存储数据,管理数据的类。开发者可以使用AppStorageV2来存储全局UI状态变量数据。它提供的是应用级的全局共享能力,开发者可以通过connect绑定同一个key,进行跨ability数据共享。 概述 AppStorageV2是一个单例,创建时间是应用UI启动时。其目的…

打靶日记 zico2: 1

一、探测靶机IP&#xff08;进行信息收集&#xff09; 主机发现 arp-scan -lnmap -sS -sV -T5 -p- 192.168.10.20 -A二、进行目录枚举 发现dbadmin目录下有个test_db.php 进入后发现是一个登录界面&#xff0c;尝试弱口令&#xff0c;结果是admin&#xff0c;一试就出 得到加…

使用Java基于Geotools的SLD文件编程式创建与磁盘生成实战

前言 在地理信息系统&#xff08;GIS&#xff09;领域&#xff0c;地图的可视化呈现至关重要&#xff0c;而样式定义语言&#xff08;SLD&#xff09;文件为地图元素的样式配置提供了强大的支持。SLD 能够精确地定义地图图层中各类要素&#xff08;如点、线、面、文本等&#x…

kubernetes》》k8s》》Service

Kubernetes 中的 Service 是用于暴露应用服务的核心抽象&#xff0c;为 Pod 提供稳定的访问入口、负载均衡和服务发现机制。Service在Kubernetes中代表了一组Pod的逻辑集合&#xff0c;通过创建一个Service&#xff0c;可以为一组具有相同功能的容器应用提供一个统一的入口地址…

【HDFS】EC重构过程中的校验功能:DecodingValidator

一、动机 DecodingValidator是在HDFS-15759中引入的一个用于校验EC数据重构正确性的组件。 先说下引入DecodingValidator的动机,据很多已知的ISSUE(如HDFS-14768, HDFS-15186, HDFS-15240,这些目前都已经fix了)反馈, EC在重构的时候可能会有各种各样的问题,导致数据错误…

现代c++获取linux系统架构

现代c获取linux系统架构 前言一、使用命令获取系统架构二、使用c代码获取系统架构三、验证四、总结 前言 本文介绍一种使用c获取linux系统架构的方法。 一、使用命令获取系统架构 linux系统中可以使用arch或者uname -m命令来获取当前系统架构&#xff0c;如下图所示 archuna…

didFinishLaunching 与「主线程首次 idle」, 哪个是更优的启动结束时间点 ?

结论先行 在这两个候选时间点里—— application:didFinishLaunchingWithOptions: 执行结束主线程第一次进入 idle&#xff08;RunLoop kCFRunLoopBeforeWaiting&#xff09; 若你只能二选一&#xff0c;以「主线程首次 idle」作为 启动结束 更合理。它比 didFinishLaunchin…