了解 React 应用程序中的渲染和重新渲染:它们如何工作以及如何优化它们

devtools/2024/10/22 8:19:37/

当我们在 react 中创建应用程序时,我们经常会遇到术语“渲染”和“重新渲染组件”。虽然乍一看这似乎很简单,但当涉及不同的状态管理系统(如 usestate、redux)或当我们插入生命周期钩子(如 useeffect)时,事情会变得有趣。如果您希望您的应用程序快速高效,那么了解这些流程是关键。

什么是渲染?

渲染是 react 根据状态或属性在屏幕上渲染用户界面 (ui) 的过程。当你的组件第一次渲染时,它被称为第一次渲染。

初始渲染如何工作?

当组件首次“安装”到 dom 时,会发生以下情况:

1。状态初始化:
无论你使用 usestate、props 还是 redux,都会创建组件的初始状态。

2。渲染函数:
react 循环遍历 jsx 代码并根据当前状态生成虚拟 dom。

3。为组件的当前状态创建一个虚拟 dom(虚拟 dom)。

4。比较(差异):
虚拟 dom 与真实 dom 进行比较(由于是第一次渲染,所以所有元素都会完全渲染)。

5。显示:
该组件显示在屏幕上。
组件渲染完成后,下一个挑战就是渲染它。

重新渲染:何时以及为何?

每次状态或道具发生变化时都会发生重新渲染。您是否单击了更改屏幕上数字的按钮?更改了 redux 存储中的值?所有这些操作都可能导致 react 再次渲染组件,这就是重新渲染的用武之地。

重新渲染如何工作?

状态变化检测:

  • 使用 usestate,当你调用 setstate 时,react 知道它需要更新组件。

  • 使用 redux,当存储中的值发生更改时,与该状态部分关联的每个组件都会重新渲染。

渲染触发器:

当状态发生变化时,react 会根据该变化创建一个新的虚拟 dom。

比较(差异):

  • react 将新的虚拟 dom 与旧的虚拟 dom 进行比较,并计算要应用哪些更改。这是 react 优化渲染的一种方式。

查看更改:

  • 计算出更改后,react 将它们应用到实际的 dom 上。因此,仅再次显示页面更改的部分。

渲染哪些组件?

并非所有组件都会受到每次更改的影响。 react 仅重新渲染那些满足以下条件的组件:

使用当地州:
如果您使用 usestate,则每次调用 setstate.

时都会重新渲染组件

使用 redux 状态:
如果你的组件依赖于 redux 状态(通过 useselector 或 connect),当该部分状态发生变化时,它将重新渲染。

使用道具:
如果 props 值发生变化,组件将使用新值重新渲染。

渲染优化

当然,不必要地重新渲染所有组件并不总是理想的。如果我们希望应用程序快速高效地运行,这里有一些优化技巧:

1。组件记忆
react 通过 react.memo 提供组件记忆功能。如果你的组件不依赖于 props 或状态变化,你可以“记住”它,因此只有当相关值发生变化时它才会重新渲染。

示例:

const memoizedcomponent = react.memo(mycomponent);

 

2。函数和值的记忆

为了避免在每次渲染时重新创建函数或值,请使用 usecallback 来记忆函数并使用 usememo 来记忆值。

  • usecallback 允许您记住函数并防止在依赖项发生变化之前重新创建它。

  • usememo 会记住函数的结果,因此不会在每次渲染时重新计算。

示例:

const increment = usecallback(() => {setcount(prevcount => prevcount + 1);
}, []);const expensivecalculation = usememo(() => {return count * 2;
}, [count]);

 

3。 redux 优化

如果您使用 redux,您可以使用记忆选择器(例如重新选择)进一步优化您的应用程序。这可以避免重新渲染不受状态更改影响的组件。

生命周期挂钩和渲染

在经典的react类中,我们使用shouldcomponentupdate来控制组件何时重新渲染。在功能组件中,可以使用 useeffect 和 memoization 来模拟这个概念。

结论

渲染和重新渲染对于 react 应用程序中的 ui 显示至关重要,但正确理解和优化这些过程可以区分慢速应用程序和超快应用程序。正确使用 memoization、usecallback、usememo 以及仔细处理 redux,有助于避免不必要的重新渲染,并保持我们的应用程序快速响应。

代码示例:实际渲染和重新渲染
这是一个使用 usestate、redux 和 memoization 来优化渲染的示例组件:

import React, { useState, useEffect, useCallback, useMemo } from 'react';
import { useSelector, useDispatch } from 'react-redux';const MyComponent = () =&gt; {// Lokalni stateconst [count, setCount] = useState(0);// Redux stateconst reduxValue = useSelector(state =&gt; state.someValue);const dispatch = useDispatch();// Memoizacija funkcije kako bi se izbeglo ponovno kreiranje na svakom renderuconst increment = useCallback(() =&gt; {setCount(prevCount =&gt; prevCount + 1);}, []);// Memoizacija izračunate vrednosticonst expensiveCalculation = useMemo(() =&gt; {return count * 2;}, [count]);// Efekat koji se pokreće samo pri promeni reduxValueuseEffect(() =&gt; {console.log("Redux value changed:", reduxValue);}, [reduxValue]);return (<div><p>Count: {count}</p><p>Expensive Calculation: {expensiveCalculation}</p><button onclick="{increment}">Increment</button><button onclick="{()"> dispatch({ type: 'SOME_ACTION' })}&gt;Dispatch Redux Action</button></div>);
};

 正如我们所看到的,这里使用了本地状态、redux、memoization 和 useeffect hook 的组合,以使应用程序尽可能高效。


http://www.ppmy.cn/devtools/115072.html

相关文章

js 深入理解类-class

目录 概述1. 类的定义2. 类构造函数2.1. 实例化2.1.1 实例化流程2.1.2 带参实例化2.1.3 执行构造函数返回的两种对象2.1.4 类构造函数和普通构造函数的区别 2.2 把类当成特殊函数2.2.1 辨别是不是函数&#xff0c;使用 typeof2.2.2 辨别是不是函数&#xff0c;是否有prototype2…

OpenCV_图像旋转超详细讲解

图像转置 transpose(src, dst); transpose()可以实现像素下标的x和y轴坐标进行对调&#xff1a;dst(i,j)src(j,i)&#xff0c;接口形式 transpose(InputArray src, // 输入图像OutputArray dst, // 输出 ) 图像翻转 flip(src, dst, 1); flip()函数可以实现对图像的水平翻转…

MySQL数据库备份与恢复

前言 数据库备份与恢复是数据库管理中的关键操作&#xff0c;它们确保了数据的安全性和在发生故障时的快速恢复。MySQL&#xff0c;作为广泛使用的开源关系型数据库管理系统&#xff0c;提供了多种备份和恢复策略。本文将详细介绍MySQL数据库的备份与恢复方法&#xff0c;包括…

记录Zabbix监控飞塔防火墙HA状态异常

1、Zabbix监控飞塔HA状态值0为正常&#xff0c;1为异常。 2、监控到HA不同步是由于防火墙策略修改导致&#xff0c;具体说明如下&#xff1a; 设备HA部署&#xff0c;当修改防火墙策略时&#xff0c;仅在一台设备上添加、删除或修改&#xff0c;该操作会导致防火墙…

软件测试工程师面试整理-编程与自动化

在软件测试领域,编程与自动化是提升测试效率、覆盖率和可靠性的关键因素。掌握编程技术和自动化测试框架,能够帮助测试人员有效地执行大量重复性测试任务,并迅速反馈软件的质量状况。以下是编程与自动化在测试中的主要应用及相关技术介绍: 1. 编程语言与自动化 ● 常用编程…

力扣300-最长递增子序列(Java详细题解)

题目链接&#xff1a;300. 最长递增子序列 - 力扣&#xff08;LeetCode&#xff09; 前情提要&#xff1a; 本题是子序列问题的开始篇&#xff0c;接下来我将更新子序列篇章的dp问题。 因为本人最近都来刷dp类的题目所以该题就默认用dp方法来做。 dp五部曲。 1.确定dp数组…

fastadmin 部署后前台会员中心出现404错误

访问前台会员中心出现404错误。 解决&#xff1a;nginx访问站点增加伪静态 location / {if (!-e $request_filename){rewrite ^(.*)$ /index.php?s$1 last; break;} }在phpstydy中增加伪静态&#xff0c;如图&#xff1a;

高可用web集群面经:集群搭建、nginx+keepalived高可用、prometheus+zabbix监控、nfs+dns

高可用web集群面经&#xff1a;集群搭建、nginxkeepalived高可用、prometheuszabbix监控、nfsdns 高可用web集群面经飞书在线链接&#x1f517;&#xff1a; (https://h03yz7idw7.feishu.cn/wiki/Ucj1wWZCGiqR68kripMcC2CLnvd)