性能:React 实战优化技巧 之 函数闭包
子组件使用了 React.memo
,为什么 “prop 值未发生改变”,子组件依然被重新渲染了?
🚧 示例:点击子组件中按钮,获取 input 数据进行提交(常见于表单)
index.tsx
import Author from './Author.tsx';export default function Index() {const [val, setVal] = useState('');const consoleValue = () => {console.log(val);}return (<><input type="text" value={val} onChange={(e) => setVal(e.target.value)} /><Author name="李刚" onClick={consoleValue} /></>);
}
Author.tsx
export default function Author({ name, onClick }: { name: string; onClick: () => void }) {console.log('Author');return (<div><button onClick={onClick}>{name}</button></div>);
}
🐋 现象:input 每次输入值,<Author>
组件就被重新渲染一次【prop onClick 发生了变化】!
🐒 使用 memo
,在 props 没有改变的情况下跳过重新渲染!
index.tsx
const AuthorMemo = React.memo(Author);export default function Index() {const consoleValue = useCallback(() => {console.log(val);}, [val]);return (<>{/* 省略 */}<AuthorMemo name="李刚" onClick={consoleValue} /></>);
}
这里需要使用 useCallback(fn, dependencies)
来处理渲染期间传递普通函数,避免传递给组件的 props 始终不同!
因为 <Author>
中要获取最新的 val
值,因此 useCallback()
中追加了 val
作为 dependencies
。
每次 <input>
输入,val
值都发生变化、从而导致 consoleValue
重新生成,因此 <AuthorMemo>
依然每次会重新渲染!
1️⃣ 传递依赖项数组: 初始渲染后以及依赖项变更后 运行
const consoleValue = useCallback(() => {// ...
}, [val]); // val 变更时返回一个新的函数
2️⃣ 传递空依赖项数组:仅在 初始渲染后 运行
const consoleValue = useCallback(() => {// ...
}, []); // 初始化后,不会再执行
3️⃣ 不传递依赖项数组:每次渲染之后 运行
const consoleValue = useCallback(() => {// ...
}); // 每一次都返回一个新函数:没有依赖项数组
🐇 上述问题应该如何解呢?
【有缺陷】方案一:memo
支持自定义 arePropsEqual
来确定是否重新渲染!
const MemoizedComponent = memo(SomeComponent, arePropsEqual?)
const AuthorMemo = React.memo(Author, (_prevProps, nextProps) => {/*** 可选参数 arePropsEqual:一个函数,接受两个参数:组件的前一个 props 和新的 props。* 如果旧的和新的 props 相等,即组件使用新的 props 渲染的输出和表现与旧的 props 完全相同,则它应该返回 true。否则返回 false。* 通常情况下,你不需要指定此函数。默认情况下,React 将使用 Object.is 比较每个 prop。*/return _prevProps.name === nextProps.name;
});export default function Index () {const consoleValue = useCallback(() => {console.log(val);}, [val]);
}
该方案的问题:获取的 val
值为一直为初始值,无法获取输入的最终 val
值。
🐇 问题分析:典型的 React 中的闭包问题。
React 中:组件内的每个函数都是一个闭包,因为组件本身只是一个函数。
理论上,当 val
发生变化时,consoleValue
函数会被重新创建,从而捕获最新的 val
值。然而,如果 AuthorMemo
没有重新渲染,或者 Author
组件内部没有正确处理 onClick
的更新【React.memo
比较算法导致 _prevProps.name === nextProps.name;
】,可能会导致 consoleValue
没有捕获到最新的 val
值。
【推荐】方案二:ref
+ useEffect
组合实现。
借助 ref 每次渲染间存储信息及修改不会触发渲染的特性;
const AuthorMemo = React.memo(Author);export default function Index() {const [val, setVal] = useState('');const ref = useRef<(() => void) | undefined>();// 省略 dependencies 参数,则在每次重新渲染组件之后,将重新运行 Effect 函数useEffect(() => {// 每次更新,指向一个新的函数// .current 属性可以随时被更新,因此它不会受到闭包的限制ref.current = () => {console.log(val);};});// 依赖数组为空 [] 在整个组件的生命周期中只会被创建一次(初始化)const consoleValue = useCallback(() => {ref.current?.();}, []);return (<><input type="text" value={val} onChange={(e) => setVal(e.target.value)} /><AuthorMemo name="李刚" onClick={consoleValue} /></>);
}
ref
用于渲染之间 存储信息(普通对象存储的值每次渲染都会重置);useEffect(() => {})
每次渲染执行;ref.current = ...
改变ref.current
属性时,React 不会重新渲染组件;
const consoleValue = useCallback(() => {}, [])
只初始渲染运行、确保了consoleValue
不发生变化(useCallback 在多次渲染中缓存函数)。
useState(initialState) useRef(initialValue)
initialState
:这个参数在首次渲染后被忽略。
🐇 原理分析:为什么没有闭包问题
- 为了让函数能够访问最新状态,每次重新渲染时都需要重新创建函数,这是无法避免的,这也是闭包的本质,与
React
无关; - 利用
Ref
是一个可变对象这一特性,从而摆脱 “过期闭包” 的问题。我们可以在过期闭包之外更改ref.current
,然后在闭包之内访问它,就可以获取最新的数据。
通过 useRef
和 useEffect
动态更新引用的函数,避免了闭包问题。consoleValue
函数虽然在整个组件生命周期中保持不变,但它通过调用 ref.current
来间接访问最新的 val
值。
1️⃣ 传递依赖项数组: 初始渲染后以及依赖项变更的重新渲染后 运行
useEffect(() => {// ...
}, [a, b]); // 如果 a 或 b 不同则会再次运行
2️⃣ 传递空依赖项数组:仅在 初始渲染后 运行
useEffect(() => {// ...
}, []); // 不会再次运行(开发环境下除外)
3️⃣ 不传递依赖项数组:每次渲染之后 运行
useEffect(() => {// ...
}); // 总是再次运行
相关文章:
性能:React 实战优化技巧 之 函数闭包
子组件使用了 React.memo ,为什么 “prop 值未发生改变”,子组件依然被重新渲染了? 🚧 示例:点击子组件中按钮,获取 input 数据进行提交(常见于表单) index.tsx import Author f…...
蓝桥杯学习大纲
(致酷德与热爱算法、编程的小伙伴们) 在查阅了相当多的资料后,发现没有那篇博客、文章很符合我们备战蓝桥杯的学习路径。所以,干脆自己整理一篇,欢迎大家补充! 一、蓝桥必备高频考点 我们以此为重点学习…...
Windows11切换回Windows10风格右键菜单
参考文章:Win11新版右键菜单用不惯?一键切换回Win10经典版!-CSDN博客 以管理员权限运行命令行cmd 切换为经典旧版右键菜单,执行 reg.exe add “HKCU\Software\Classes\CLSID\{86ca1aa0-34aa-4e8b-a509-50c905bae2a2}\InprocServe…...
Python 爬虫selenium
1.selenium自动化 selenium可以操作浏览器,在浏览器页面上实现:点击、输入、滑动 等操作。 不同于selenium自动化,逆向本质是: 分析请求,例如:请求方法、请求参数、加密方式等。用代码模拟请求去实现同等…...
Linux常用操作
软件安装 CentOS系统使用: yum [install remove search] [-y] 软件名称 install 安装 remove 卸载 search 搜索 -y,自动确认 Ubuntu系统使用 apt [install remove search] [-y] 软件名称 install 安装 remove 卸载 search 搜索 -y,自动确认 yum 和 apt 均需要root权限 syste…...
Note25022001_Excel表格如何在文字的中间或者后边插入当前日期
Excel表格如何在文字的中间或者后边插入当前日期 关键字: EXCEL;当前日期;文字中间 如图所示: 其中一种实现方法如下: 打开表格,在某一个表格中输入: "项目计划管理表(厂内&…...
Django5 实用指南(四)URL路由与视图函数
4.1 Django5的URL路由系统 Django 的 URL 路由系统是其核心组件之一,它负责将用户的 HTTP 请求(即 URL)映射到相应的视图函数上。每当用户在浏览器中访问某个 URL 时,Django 会根据项目的 URL 配置文件(urls.py&#…...
Unity3D 基于 GPU 动画和 Compute Shader 的大批量动画渲染详解
引言 在现代游戏开发中,渲染大量动画角色是一个常见的需求,尤其是在大规模战斗场景、开放世界游戏或 VR/AR 应用中。传统的 CPU 动画计算和渲染方式在面对大批量角色时,往往会遇到性能瓶颈。为了优化性能,开发者可以利用 GPU 的强…...
遥感影像目标检测:从CNN(Faster-RCNN)到Transformer(DETR)
我国高分辨率对地观测系统重大专项已全面启动,高空间、高光谱、高时间分辨率和宽地面覆盖于一体的全球天空地一体化立体对地观测网逐步形成,将成为保障国家安全的基础性和战略性资源。未来10年全球每天获取的观测数据将超过10PB,遥感大数据时…...
什么是DeFi (去中心化金融)
DeFi (去中心化金融) 概述 💰 1. DeFi 基础概念 1.1 什么是 DeFi? DeFi 是建立在区块链上的金融服务生态系统,它: 无需中心化中介开放且透明无需许可即可参与代码即法律 1.2 DeFi 的优势 开放性:任何人都可以参与…...
深入解析 sudo -l 命令的输出内容
在 Linux 系统中,sudo 命令允许普通用户以超级用户(root)权限执行命令。sudo -l 命令用于查看当前用户在 sudoers 配置文件中的权限,以及与 sudo 相关的安全策略。本文将详细解析 sudo -l 输出的各个部分,包括 用户权限…...
DeepSeek 助力 Vue 开发:打造丝滑的瀑布流布局(Masonry Layout)
前言:哈喽,大家好,今天给大家分享一篇文章!并提供具体代码帮助大家深入理解,彻底掌握!创作不易,如果能帮助到大家或者给大家一些灵感和启发,欢迎收藏关注哦 💕 目录 Deep…...
linux学习【7】Sourc Insight 4.0设置+操作
目录 1.Source Insight是什么?2.需要哪些配置?3.怎么新建项目4.一些问题的解决1.中文乱码问题 按照这个设置就可以了,下面的设置会标明设置理由。 1.Source Insight是什么? 阅读源码,编辑源码,不能编译&am…...
OpenCV形态学操作
1.1. 形态学操作介绍 初识: 形态学操作是一种基于图像形状的处理方法,主要用于分析和处理图像中的几何结构。其核心是通过结构元素(卷积核)对图像进行扫描和操作,从而改变图像的形状和特征。例如: 腐蚀&…...
使用Python中的`gensim`库构建LDA(Latent Dirichlet Allocation)模型来分析收集到的评论
下面为你详细介绍如何使用Python中的gensim库构建LDA(Latent Dirichlet Allocation)模型来分析收集到的评论。LDA是一种主题模型,它可以将文档集合中的文本按照主题进行分类。 步骤概述 数据预处理:对收集到的评论进行清洗、分词…...
【STM32】外部时钟|红外反射光电开关
1.外部时钟 单片机如何对外部触发进行计数?先看一下内部时钟,内部时钟是接在APB1和APB2时钟线上的,APB1,APB2来自stm32单片机内部的脉冲信号,也叫内部时钟。我们用来定时。同样我们可以把外部的信号接入单片机,来对其…...
Visual Studio Code 集成 Baidu Comate
文章目录 安装Baidu Comate插件 安装Baidu Comate插件 从左主侧栏中 点击 【扩展】这个图标,然后在上方输入栏中输入 baidu comate —>选中列出的Bai Comate —>点击 【安装】按钮,等待安装完毕…...
数据结构者
数据(data):可被计算机接受处理的符号总称 数据元素(data element):数据的基本单位,常作为一个整体进行考虑和处理 一个数据元素可以由若干个数据项(data item)组成 数…...
论文阅读 DOES END-TO-END AUTONOMOUS DRIVING REALLY NEED PERCEPTION TASKS?
端到端的强势来袭,好久了~~~ 简单翻译:端到端真的需要感知任务嘛? code https://github.com/PeidongLi/SSR. https://arxiv.org/pdf/2409.18341 1. 摘要 端到端自动驾驶(E2EAD)方法通常依赖于监督式感知任务来提取显…...
总结UML类图几种关系画法
最近在公司需要画类图,有些遗忘,于是重新学习并并总结了一下: 继承:实线空心三角箭头。A继承B,A指向B。A是一种B。实现:虚线空心三角箭头。A实现B,A指向B。A实现B接口。关联:实线普…...
python-leetcode 40.二叉树的层序遍历
题目: 给定二叉树的根节点root,返回其节点值得层序遍历(即逐层从左到右访问所有节点) 方法:广度优先搜索 # Definition for a binary tree node. # class TreeNode(object): # def __init__(self, val0, leftNone, rightNon…...
安利:外文文献翻译插件
沉浸式翻译 好用吹爆!!!!!!!! 任务 2.谷歌学术 复制外文的标题搜索 3.进入文献所在,登录对应机构 4.可以看到都翻译好了 5.然后复制到文档,改改格式巴…...
Python——生成AIGC图像
文章目录 一、背景介绍 二、效果图展示 三、完整代码 四、分步解释 五、实用建议 1)提示词技巧 2)性能优化 3)常见问题处理 4)扩展功能建议 六、注意事项 1. 硬件要求 2. 法律合规 3. 模型安全 一、背景介绍 AIGC&a…...
OpenGL(2)基于Qt做OpenGL开发
文章目录 一、基于Qt做OpenGL开发1、环境准备2、创建OpenGL窗口3、绘制基本图形 一、基于Qt做OpenGL开发 1、环境准备 确保你已经安装了 Qt 开发环境(包含 Qt Creator),并且支持 OpenGL 开发。在创建 Qt 项目时,选择 “Qt Widget…...
【Windows软件 - HeidiSQL】导出数据库
HeidSQL导出数据库 软件信息 具体操作 示例文件 选项分析 选项(1) 结果(1) -- -------------------------------------------------------- -- 主机: 127.0.0.1 -- 服务器版本: …...
用deepseek学大模型04-模型可视化与数据可视化
deepseek.com: pytorch可视化工具 生成神经网络图 在 PyTorch 中,可视化神经网络结构的常用工具和方法有以下几种,以下将详细介绍它们的用法: 1. TensorBoard (PyTorch 官方集成) PyTorch 通过 torch.utils.tensorboard 支持 TensorBoard&a…...
rust笔记8-Deref与隐式解引用强制转换
Rust 的智能指针和 Deref Trait 是 Rust 中非常重要的概念,它们使得 Rust 的引用和指针操作更加灵活和安全。下面我们将深入介绍 Deref Trait、Deref 与 &、* 运算符的关系,以及 Rust 的隐式解引用强制转换(Deref Coercion)。 1. 智能指针与 Deref Trait 智能指针(如…...
Debian软件包重构
Explore projects GitLab 1. apt-get build-dep <pkg> ## 安装编译依赖包 2. apt source <pkg> ## 下载 <pkg> 包的源码 3. 创建 git ,打补丁 4. dpkg-buildpackage -b -uc -us -d ## 重新打包编译 # 解压出包中的文件到 extract 目…...
学习 `@PreDestroy`:Java EE/Jakarta EE 生命周期回调
学习 PreDestroy:Java EE/Jakarta EE 生命周期回调 前言1. 什么是 PreDestroy?2. PreDestroy 的用途3. 使用 PreDestroy 的条件4. 代码示例运行结果: 5. PreDestroy 的调用时机6. 注意事项7. 实际应用场景场景 1:数据库连接管理场…...
JDK最详细安装教程,零基础入门到精通,收藏这篇就够了
目录 一、下载与安装二、配置环境三、验证是否配置成功 一、下载与安装 1、下载地址 http://www.oracle.com/technetwork/java/javase/downloads/index.html 2、选择自己想要的版本下载,并且选择自己电脑对应的版本下载 3、下载完成之后,双击打开然后…...
解决DeepSeek服务器繁忙问题的实用指南
目录 简述 1. 关于服务器繁忙 1.1 服务器负载与资源限制 1.2 会话管理与连接机制 1.3 客户端配置与网络问题 2. 关于DeepSeek服务的备用选项 2.1 纳米AI搜索 2.2 硅基流动 2.3 秘塔AI搜索 2.4 字节跳动火山引擎 2.5 百度云千帆 2.6 英伟达NIM 2.7 Groq 2.8 Firew…...
【一个人的第一年】成都Java日志7
#海纳百川,有容乃大# 1.互联网 互联网是一个服务我的工具而不是绑架任何人的场所,网络不是现实,现实才是生活。 现在越来越多的声音出现在互联网,能轻易地挑起群体情绪。对其他的任何事可以有不同的立场和观点,而不是…...
回调处理器
文章目录 什么是回调处理器回调处理器的工作流程回调处理器的使用自定义链组件中的回调 内置回调处理器自定义回调处理器 在编程领域中,回调是一个非常重要的概念。简而言之,回调是一种特殊的函数或方法,它可以被传递给另一个函数作为参数&am…...
国产编辑器EverEdit -告别东找西找!一键打开当前文件所在目录!
1 文件操作 2 应用场景 在文件编辑过程中,有时需要对文件进行一些操作,比如:在命令窗口输入文件路径、文件名,进入到文件目录,对文件进行压缩等,如果没有直达命令,用户需要通过文件管理器找到目…...
23种设计模式 - 解释器模式
模式定义 解释器模式(Interpreter Pattern)是一种行为型设计模式,用于为特定语言(如数控系统的G代码)定义文法规则,并构建解释器来解析和执行该语言的语句。它通过将语法规则分解为多个类,实现…...
深度学习在语音识别中的应用
引言 语音识别技术是人工智能领域中的一个重要分支,它使得机器能够理解和转换人类的语音为文本。深度学习的出现极大地推动了语音识别技术的发展。本文将介绍如何使用深度学习构建一个基本的语音识别系统,并提供一个实践案例。 环境准备 在开始之前&a…...
【相聚青岛】人工智能与材料国际学术会议即将召开
一、大会简介 人工智能与材料国际会议(ICAIM 2025) 官方网站:www.ic-aim.net 官方邮箱:icaim2025163.com 会议时间:2025年3.21-24 会议地点:中国青岛 会议检索:EI检索 截稿时间:2月…...
PHP会务会议系统小程序源码
📅 会务会议系统 一款基于ThinkPHPUniapp框架,精心雕琢的会议管理微信小程序,专为各类高端会议场景量身打造。它犹如一把开启智慧殿堂的金钥匙,为会议流程优化、开支精细化管理、数量精准控制、标准严格设定以及供应商严格筛选等…...
能够私有化部署的集装箱箱号识别软件,技术实现方法
启智畅想集装箱箱号识别软件是一种基于计算机视觉和OCR(光学字符识别)技术的工具,主要用于自动识别集装箱上的唯一编号(如ISO6346标准的箱号,格式为4位字母7位数字1位校验码)。以下是关于此类软件的关键信息…...
idea连接gitee后.反向创建仓库和分支
文章目录 自动关联你登录的账号填写你的仓库和分支完成后会在gitee上创建一个仓库 (使用idea远程兼容gitee并反向创建仓库和分支) 自动关联你登录的账号 填写你的仓库和分支 完成后会在gitee上创建一个仓库...
Java——面向对象编程
面向对象编程(Object-Oriented Programming, OOP)是Java的核心特性之一。它通过将现实世界中的事物抽象为对象,使程序更易于理解、维护和扩展。以下是Java面向对象编程的详细介绍: 1. 面向对象编程的四大特性 Java的面向对象编程…...
MAC快速本地部署Deepseek (win也可以)
MAC快速本地部署Deepseek (win也可以) 下载安装ollama 地址: https://ollama.com/ Ollama 是一个开源的大型语言模型(LLM)本地运行框架,旨在简化大模型的部署和管理流程,使开发者、研究人员及爱好者能够高效地在本地环境中实验和…...
从猜想终结到算法革新,弹性哈希开启数据存储新篇章
目录 哈希表的前世今生基本原理从传统到现代:哈希表的演变历程 安德鲁 克拉皮文及其团队的创作历程弹性哈希详解基本原理优点技术细节 漏斗哈希解析基本原理优点技术细节 新算法的实际应用案例电子商务推荐系统金融交易监控系统社交媒体内容过滤物联网设备管理 结论…...
云端SaaS系统架构设计
随着互联网的发展,SaaS(软件即服务)架构在众多行业中得到了广泛应用。作为一种高效、可扩展的服务模式,SaaS不仅提升了企业的信息化水平,也使得服务提供商能够通过云计算平台实现全球范围内的业务交付。在设计一个现代…...
黑盒测试和白盒测试的主要优缺点
黑盒测试 vs. 白盒测试:优缺点对比 类别黑盒测试(Black-box Testing)白盒测试(White-box Testing)定义不关注代码实现,仅测试功能和输入输出关注代码逻辑,测试代码内部实现测试依据需求文档、功…...
获取每月最后一个工作日:考虑法定节假日与调休
在许多业务场景中,了解每个月的最后一个工作日对于财务结算、报告生成等至关重要。然而,确定这一日期时必须考虑到国家的法定节假日以及可能存在的调休安排。本文将介绍如何通过Java编写一个工具类来获取指定月份的最后一个工作日,并利用第三…...
Hayabusa:一款针对Windows事件日志的威胁搜索与取证分析工具
关于Hayabusa Hayabusa是一款针对Windows事件日志的威胁搜索与取证分析工具,该工具基于内存安全的Rust开发,支持多线程运行,并且是唯一完全支持 Sigma 规范(包括 v2 关联规则)的开源工具。 Hayabusa本质上是一个Windo…...
leetcode_位运算 190.颠倒二进制位
190. 颠倒二进制位 颠倒给定的 32 位无符号整数的二进制位。 1. 字符串 class Solution:# param n, an integer# return an integerdef reverseBits(self, n):res "" # 创建一个保存结果的空字符串for b in str(bin(n))[2:]:# 遍历n的二进制数res b res # 把每…...
O1 Embedder:让检索器思考后再行动
25年2月来自中科大和北京智源研究院的论文“O1 Embedder: Let Retrievers Think Before Action”。 大语言模型 (LLM) 的功能日益强大,彻底改变人们获取和利用信息的方式。值得注意的是,LLM 擅长执行细粒度数据表示,这有助于精确检索信息。它…...
Python项目中一些常用的关键字
数据定义与类型相关 int:表示整数类型。 float:用于表示浮点数类型。 str:字符串类型,用于处理文本数据。 bool:布尔类型,取值为True或False。 list:列表类型,可存储多个元素&am…...