React -- memo允许你的组件在 props 没有改变的情况下跳过重新渲染。

memo(Component, arePropsEqual?) 

使用 memo 将组件包装起来,以获得该组件的一个 记忆化 版本。通常情况下,只要该组件的 props 没有改变,这个记忆化版本就不会在其父组件重新渲染时重新渲染。但 React 仍可能会重新渲染它:记忆化是一种性能优化,而非保证。

import { memo } from 'react';const SomeComponent = memo(function SomeComponent(props) {// ...
});

参数

  • Component:要进行记忆化的组件。memo 不会修改该组件,而是返回一个新的、记忆化的组件。它接受任何有效的 React 组件,包括函数组件和 forwardRef 组件。

  • 可选参数 arePropsEqual:一个函数,接受两个参数:组件的前一个 props 和新的 props。如果旧的和新的 props 相等,即组件使用新的 props 渲染的输出和表现与旧的 props 完全相同,则它应该返回 true。否则返回 false。通常情况下,你不需要指定此函数。默认情况下,React 将使用 Object.is 比较每个 prop。

返回值
memo 返回一个新的 React 组件。它的行为与提供给 memo 的组件相同,只是当它的父组件重新渲染时 React 不会总是重新渲染它,除非它的 props 发生了变化。

用法

当 props 没有改变时跳过重新渲染
React 通常在其父组件重新渲染时重新渲染一个组件。你可以使用 memo 创建一个组件,当它的父组件重新渲染时,只要它的新 props 与旧 props 相同时,React 就不会重新渲染它。这样的组件被称为 记忆化的(memoized)组件。

要记忆化一个组件,请将它包装在 memo 中,使用它返回的值替换原来的组件:

const Greeting = memo(function Greeting({ name }) {return <h1>Hello, {name}!</h1>;
});export default Greeting;

React 组件应该始终具有 纯粹的渲染逻辑。这意味着如果其 props、state 和 context 没有改变,则必须返回相同的输出。通过使用 memo,你告诉 React 你的组件符合此要求,因此只要其 props 没有改变,React 就不需要重新渲染。即使使用 memo,如果它自己的 state 或正在使用的 context 发生更改,组件也会重新渲染。

在此示例中,请注意 Greeting 组件在 name 更改时重新渲染(因为那是它的 props 之一),但是在 address 更改时不会重新渲染(因为它不作为 props 传递给 Greeting):

import { memo, useState } from 'react';export default function MyApp() {const [name, setName] = useState('');const [address, setAddress] = useState('');return (<><label>Name{': '}<input value={name} onChange={e => setName(e.target.value)} /></label><label>Address{': '}<input value={address} onChange={e => setAddress(e.target.value)} /></label><Greeting name={name} /></>);
}const Greeting = memo(function Greeting({ name }) {console.log("Greeting was rendered at", new Date().toLocaleTimeString());return <h3>Hello{name && ', '}{name}!</h3>;
});

更改Address,Geeting没有重渲染。
在这里插入图片描述
更改name,Greeting重新渲染了
在这里插入图片描述
在这里插入图片描述

在每个地方都应该添加 memo 吗?

如果你的应用像此站点一样,大多数交互是粗略的(例如直接替换页面或整个部分),那么通常不需要记忆化。另一方面,如果你的应用更像是绘图编辑器,大多数交互是细粒度的(例如移动图形),那么你可能会发现记忆化非常有用。

只有当你的组件经常使用完全相同的 props 重新渲染时,并且其重新渲染逻辑是非常昂贵的,使用 memo 优化才有价值。如果你的组件重新渲染时没有明显的延迟,那么 memo 就不必要了。请记住,如果传递给组件的 props 始终不同,例如在渲染期间传递对象或普通函数,则 memo 是完全无用的。这就是为什么你通常需要在 memo 中同时使用 useMemo useCallback

在其他情况下将组件包装在 memo 中是没有任何好处的。这种做法也没有什么明显的危害,因此一些团队会选择不考虑个别情况,并尽可能使用 memo。这种方法的缺点是代码变得不易读。此外,并不是所有的记忆化都是有效的:一个“总是新的”值足以破坏整个组件的记忆化。

实践中,你可以通过遵循一些原则来使许多 memoization 变得不必要:

当一个组件在视觉上包裹其他组件时,让它 接受 JSX 作为子组件。这样,当包装组件更新其自身状态时,React 知道其子组件不需要重新渲染。
优先使用局部状态,并且不要将 状态提升 到不必要的层级。例如,不要将短暂状态(如表单数据和项元素是否 hover 状态)保留在树的顶部或全局状态库中。
保持你的 渲染逻辑纯粹。如果重新渲染组件会导致问题或产生一些明显的视觉瑕疵,则这是你组件中的 bug!修复 bug 而不是添加 memoization。
避免 不必要的 Effect 来更新状态。React 应用中的大多数性能问题都是由于 Effect 引起的更新链,这些 Effect 会使你的组件一次又一次地重新渲染。
尝试 从你的 Effect 中删除不必要的依赖项。例如,与其使用 memoization,不如将某些对象或函数移动到 Effect 内部或组件外部,这通常更简单。
如果特定交互仍然感觉不流畅,请 使用 React 开发者工具 profiler 来查看哪些组件最需要 memoization,并在需要时添加 memoization。这些原则使你的组件更易于调试和理解,因此建议在任何情况下都遵循它们。从长远来看,我们正在研究 自动进行细粒度 memoization,以解决这个问题。

当组件的某个 prop 是对象、数组或函数时,我的组件会重新渲染。

React 通过浅比较来比较旧的和新的 prop:也就是说,它会考虑每个新的 prop 是否与旧 prop 引用相等。如果每次父组件重新渲染时创建一个新的对象或数组,即使它们每个元素都相同,React 仍会认为它已更改。同样地,如果在渲染父组件时创建一个新的函数,即使该函数具有相同的定义,React 也会认为它已更改。为了避免这种情况,可以简化 props 或在父组件中记忆化(memoize)props。
官网讲解: memo

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

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

相关文章

内容产品运营方案业务架构解析与实践探索

### 背景 在信息爆炸的时代背景下&#xff0c;内容产品运营成为各行各业竞争的重要环节。构建合理的内容产品运营方案业务架构&#xff0c;能够帮助企业更好地管理内容生产、推广和变现&#xff0c;提升品牌影响力和商业价值。 ### 业务架构设计 #### 1. 内容生产与管理 建立…

Chrome Plugin静态页面触发CSP如何解决CSP

CSP 内容安全策略 (Content Security Policy)&#xff1a; 内容安全策略是一种计算机安全标准&#xff0c;旨在防御跨站脚本、点击劫持等代码注入攻击&#xff0c;阻止恶意内容在受信网页环境中执行。 Manifest V3 对于内容安全策略有一些默认的设置&#xff0c;如禁止外部代码…

漏洞挖掘 | 记一次信息泄露到登入后台

这次是项目上遇到的一个洞&#xff0c;打开页面是一个红红的登录页面 这里就不放图了&#xff0c;浓浓的红色气息~ 老样子抓登录包 虽然是明文传输但是爆破弱口令无果 f12大法&#xff0c;审计源代码&#xff0c;在其中一个js文件中发现了这个接口 拼接URL进行访问 感觉有点东…

Python与CMD的交织之旅:探索、挑战与突破

Python与CMD的交织之旅&#xff1a;探索、挑战与突破 在编程的世界中&#xff0c;Python以其简洁明了的语法和强大的功能库赢得了广大开发者的青睐。而CMD&#xff0c;作为Windows操作系统的命令行工具&#xff0c;也是程序员们进行各种操作的重要工具。那么&#xff0c;Pytho…

微信小程序和支付宝小程序生成二维码

描述&#xff1a;前端生成二维码 一、页面 <canvas type"2d" style"width: 312px; height: 312px;" id"myQrcode"></canvas>二、微信小程序 安装 weapp-qrcode-canvas-2d 官方文档传送门 const query wx.createSelectorQuery()…

优质域名怎么注册?

在数字化时代&#xff0c;一个优质的域名对于企业和个人品牌的重要性不言而喻。它不仅能够提升品牌形象&#xff0c;还能带来可观的流量和潜在客户。然而&#xff0c;注册一个优质域名并非易事&#xff0c;它需要策略和技巧。本文将探讨如何注册一个优质域名&#xff0c;包括选…

qgroundcontrol编程入门:探索无人机控制的新境界

qgroundcontrol编程入门&#xff1a;探索无人机控制的新境界 在无人机技术的飞速发展下&#xff0c;qgroundcontrol作为一款强大的地面控制站软件&#xff0c;逐渐成为了无人机开发者和爱好者们的首选工具。然而&#xff0c;对于初学者而言&#xff0c;qgroundcontrol的编程入…

二、electron 的 BrowserWindow 的属性

const { app, BrowserWindow } require(electron)function createWindow() {let mainWin new BrowserWindow({x: 100,y: 100, //设置窗口的显示位置&#xff0c;相对于当前屏幕的左上角 show: true, //默认情况下创建一个窗口对象之后就会显示&#xff0c;设置为false&#x…

移动性:mobility

移动性管理原理 什么是移动性 一些术语 移动性管理办法 让网络&#xff08;路由器&#xff09;处理移动性&#xff1a;路由器通过常规路由表发送交换节点的永久地址 路由器互相交换各自移动节点位置信息&#xff0c;路由系统定期会通告自己这边新移动过来的移动设备信息端系统…

Spark大数据 IDEA开发词频统计项目

在使用IntelliJ IDEA进行Spark大数据项目的开发时&#xff0c;词频统计&#xff08;Word Count&#xff09;是一个常见的入门项目。以下是一个基本的步骤指南&#xff0c;用于在IDEA中创建和运行一个Spark词频统计项目&#xff1a; 1. 设置开发环境 确保你已经安装了以下软件…

【Linux】系统优化:一键切换软件源与安装Docker

引言 在Linux系统安装完成后&#xff0c;进行一些必要的初始化设置是提升系统性能和用户体验的关键。本文将重点介绍两个实用的一键脚本&#xff1a;LinuxMirrors提供的软件源切换脚本和Docker安装脚本。这两个脚本将帮助我们简化配置安装过程。 一键切换软件源脚本 在Linux…

前端之npm运行时配置文件.npmrc(可用于配置npm淘宝源)

文章目录 前端之npm运行时配置文件.npmrc什么是.npmrc设置项目配置文件设置用户配置文件设置全局配置文件给npm 命令添加注册源选项 前端之npm运行时配置文件.npmrc 什么是.npmrc 官网&#xff1a;https://nodejs.cn/npm/cli/v7/configuring-npm/npmrc/ .npmrc&#xff0c;可…

【0286】Postgres内核 shared buffer pool 初始化实现

0. 前言 Postgres内核中shared buffer的初始化过程是在buf_init.c(缓冲区管理器初始化例程,src/backend/storage/buffer)源文件中实现的。 该文件中的几个注释值得重视: Data Structures(数据结构) 缓冲区位于freelist和查找数据结构中。 Buffer Lookup(缓存查找) 两…

QT 和VS 针对linux开发的不同

1.qt 简介 Qt是一个跨平台的C图形用户界面应用程序开发框架&#xff0c;由Qt Company开发。它最初由挪威的Trolltech公司开发&#xff0c;后被诺基亚收购&#xff0c;并在2012年再次被Digia收购。Qt框架提供了丰富的功能和工具&#xff0c;使开发者能够更快速、更高效地创建各…

【Test 58】 Qt信号与槽机制! 高频的Qt 知识点!

文章目录 1.Qt 信号与槽机制原理&#xff08;Signal & Slot&#xff09;2. QObject 类 connect 的介绍3. 信号与槽机制连接方式4. 信号和槽机制优势及其效率&#xff1a;5. 信号与槽机制应用 1.Qt 信号与槽机制原理&#xff08;Signal & Slot&#xff09; &#x1f42…

【Java笔记】第9章:三个修饰符

前言1. abstract&#xff08;抽象的&#xff09;2. static&#xff08;静态的&#xff09;3. final&#xff08;最终的&#xff09;结语 上期回顾:【Java笔记】第8章&#xff1a;面向对象的三大特性&#xff08;封装、继承、多态&#xff09; 个人主页&#xff1a;C_GUIQU 归属…

小抄 20240607

1 一定要多接触幸运的人&#xff0c;好运的人更有可能继续好运。 这不是迷信&#xff0c;好运的背后是见识、性格、逻辑的加持&#xff0c;一定有过人之处&#xff0c;才能经常好运。 反过来&#xff0c;那些经常走霉运的人&#xff0c;一定是底层逻辑出了问题&#xff0c;陷…

【GUI软件】小红书搜索结果批量采集,支持多个关键词同时抓取!

文章目录 一、背景介绍1.1 爬取目标1.2 演示视频1.3 软件说明 二、代码讲解2.1 爬虫采集模块2.2 软件界面模块2.3 日志模块 三、获取源码及软件 一、背景介绍 1.1 爬取目标 您好&#xff01;我是马哥python说 &#xff0c;一名10年程序猿。 我用python开发了一个爬虫采集软件…

Apache Doris 基础 -- 数据表设计(模式更改)

用户可以通过schema Change操作修改现有表的模式。表的模式主要包括对列的修改和对索引的修改。这里我们主要介绍与列相关的Scheme更改。对于与索引相关的更改&#xff0c;可以查看数据表设计/表索引&#xff0c;查看每个索引的更改方法。 1、术语 基本表&#xff08;Base Ta…

IMX6ULL kernel移植

1.环境说明 编译主机Debian 12交叉编译工具arm-none-linux-gnueabihf-gcc(gcc version 12.3.1 20230626)kernel版本lf-6.6.3-1.0.02.官方默认版本说明 默认官方版本的kernel是可以运行的,直接按照下面的命令编译即可 ###清除掉默认配置make ARCH=arm CROSS_COMPILE=arm-no…