」工欲善其事,必先利其器。「—孔子《論語.錄靈公》
首頁 > 程式設計 > 了解 React 應用程式中的渲染和重新渲染:它們如何運作以及如何優化它們

了解 React 應用程式中的渲染和重新渲染:它們如何運作以及如何優化它們

發佈於2024-11-03
瀏覽:453

Razumevanje Renderovanja i Rerenderovanja u React Aplikacijama: Kako funkcionišu i kako ih optimizovati

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

什么是渲染?

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

初始渲染如何工作?

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

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

2.渲染函数:
React 循环遍历 JSX 代码并根据当前状态生成虚拟 DOM。

3.为组件的当前状态创建一个虚拟 DOM(Virtual 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 = () => {
  // Lokalni state
  const [count, setCount] = useState(0);

  // Redux state
  const reduxValue = useSelector(state => state.someValue);
  const dispatch = useDispatch();

  // Memoizacija funkcije kako bi se izbeglo ponovno kreiranje na svakom renderu
  const increment = useCallback(() => {
    setCount(prevCount => prevCount   1);
  }, []);

  // Memoizacija izračunate vrednosti
  const expensiveCalculation = useMemo(() => {
    return count * 2;
  }, [count]);

  // Efekat koji se pokreće samo pri promeni reduxValue
  useEffect(() => {
    console.log("Redux value changed:", reduxValue);
  }, [reduxValue]);

  return (
    

Count: {count}

Expensive Calculation: {expensiveCalculation}

); };

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

版本聲明 本文轉載於:https://dev.to/jelena_petkovic/razumevanje-renderovanja-i-rerenderovanja-u-react-aplikacijama-kako-funkcionisu-i-kako-ih-optimizovati-40cj?1如有侵犯,請聯絡study_golang @163.com刪除
最新教學 更多>
  • 如何使用 whereBetween() 和 Carbon 來擷取 Laravel 中的重複項目?
    如何使用 whereBetween() 和 Carbon 來擷取 Laravel 中的重複項目?
    Laravel $q->where() Between Dates: An Optimized Approach for Recurring Projects檢索設定為訂續或重複的專案特定時間段,您可以將$q->where() 方法與自訂函數一起使用。然而,在 Laravel 中有更有效的方法來實現...
    程式設計 發佈於2024-11-03
  • 模擬 Python 類別
    模擬 Python 類別
    最近,我必須使用 Pytest 為 Python 模組編寫單元測試。此模組包含一個類,其他類在其建構函數中初始化。 像往常一樣,我為此類創建了一個固定裝置,以便輕鬆為每個類別方法編寫測試。此時,當我嘗試模擬建構函數中啟動的不同類別時,我遇到了一些問題。模擬不起作用,這些類別的實例仍在建立中。 經...
    程式設計 發佈於2024-11-03
  • 如何在單一 MySQL 查詢中取得多筆記錄及其相關資料?
    如何在單一 MySQL 查詢中取得多筆記錄及其相關資料?
    在一個回應中傳回多個回應資料在您的PHP 程式碼中,您有一個從資料庫表中取得多行資料的查詢。但是,您收到多個回應,並且希望將它們合併為具有多個記錄的單一回應。 要實現此目的,您需要修改查詢以對必要的表執行左聯接。您將獲取主題數據並使用左連接包含相關學生信息,而不是獲取學生數據。這將允許您在單一回應中...
    程式設計 發佈於2024-11-03
  • D - 依賴倒置原理(DIP)
    D - 依賴倒置原理(DIP)
    Before understanding DIP (Dependency Inversion Principle), it's important to know what High-Level and Low-Level modules and abstractions are. ...
    程式設計 發佈於2024-11-03
  • 為什麼 JavaScript RegEx 無法驗證輸入?
    為什麼 JavaScript RegEx 無法驗證輸入?
    Javascript 中的RegEx 功能的鬥爭:“Regex 不工作”的案例研究在標題中引用的查詢的上下文中,“Javascript RegEx 不工作” ,」使用者遇到了一個問題,即無論輸入值如何,正則表達式(regEx) 始終返回false。查詢中提供的程式碼片段如下:function che...
    程式設計 發佈於2024-11-03
  • 如何對 Ajax 請求進行排序以實現最佳控制?
    如何對 Ajax 請求進行排序以實現最佳控制?
    對Ajax 請求進行排序迭代集合並對每個元素進行單獨的Ajax 呼叫時,必須控制順序以防止伺服器過載和瀏覽器凍結。雖然可以使用自訂迭代器,但還有更優雅的解決方案可用。 jQuery 1.5 在jQuery 1.5 及更高版本中,$.ajaxQueue() 插件利用$ .Deferred、$.queu...
    程式設計 發佈於2024-11-03
  • 如何為 DOM 元素產生精確的 CSS 路徑?
    如何為 DOM 元素產生精確的 CSS 路徑?
    以增強的精度從 DOM 元素檢索 CSS 路徑提供的函數嘗試為給定 DOM 元素生成 CSS 路徑。然而,它的輸出缺乏特異性,無法捕捉元素在其兄弟元素中的位置。為了解決這個問題,我們需要一個更複雜的方法。 改進的 CSS 路徑函數下面介紹的增強函數解決了原來的限制:var cssPath = fun...
    程式設計 發佈於2024-11-03
  • 如何將單一 Python 字典寫入具有精確標題和值行的 CSV 檔案?
    如何將單一 Python 字典寫入具有精確標題和值行的 CSV 檔案?
    探索將Python 字典寫入CSV 文件的細微差別您對將Python 字典無縫寫入CSV 文件的追求給您帶來了意想不到的挑戰。雖然您設想在作為標題的字典鍵和作為第二行的值之間進行清晰的劃分,但您目前的方法似乎還不夠。讓我們深入細節,解鎖解決方案。 問題在於方法的選擇。 DictWriter.writ...
    程式設計 發佈於2024-11-03
  • 如何處理 Go 中延遲函數的錯誤回傳值?
    如何處理 Go 中延遲函數的錯誤回傳值?
    處理Go 中返回值錯誤的延遲函數當返回變數的函數在沒有延遲的情況下被延遲時,gometalinter 和errcheck 正確地發出警告檢查其回傳的錯誤。這可能會導致未處理的錯誤和潛在的運行時問題。 處理這種情況的習慣用法不是推遲函數本身,而是將其包裝在另一個檢查返回值的函數中。這是一個例子:def...
    程式設計 發佈於2024-11-03
  • 為什麼程式設計師不能總是記住程式碼:背後的科學
    為什麼程式設計師不能總是記住程式碼:背後的科學
    如果您曾經想知道為什麼程式設計師很難回憶起他們編寫的確切程式碼,那麼您並不孤單。儘管花了數小時編碼,許多開發人員經常忘記細節。這並不是因為缺乏知識或經驗,而是因為工作本身的本質。讓我們來探究一下這種現象背後的原因。 程式設計的本質 透過記憶解決問題 這比僅僅記憶語法...
    程式設計 發佈於2024-11-03
  • 你並不孤單:在社群的支持下掌握 Python
    你並不孤單:在社群的支持下掌握 Python
    加入 Python 社群可取得:社群論壇:向經驗豐富的開發者取得支援和建議(如 Stack Overflow)。 Discord 伺服器:即時聊天室,提供即時支援與指導(如 Python Discord)。線上課程與研討會:來自專家的指導,涵蓋各種主題(如 Udemy 上的 Python NumPy...
    程式設計 發佈於2024-11-03
  • 學習夥伴
    學習夥伴
    聊天機器人介面,允許使用者輸入訊息並接收來自 GPT-3.5 語言模型的對話回應。 特徵 用於處理 HTTP 請求的基於 Flask 的 Web 伺服器。 呈現用作使用者介面的基本 HTML 模板 (chat.html)。 透過 POST 請求接受使用者輸入並將其傳送到 OpenAI 的 GP...
    程式設計 發佈於2024-11-03
  • 前端開發 + 資料結構與演算法:DSA 如何為您的 React 應用程式提供動力 ⚡
    前端開發 + 資料結構與演算法:DSA 如何為您的 React 應用程式提供動力 ⚡
    专注于前端的面试通常根本不关心 DSA。 对于我们这些记得在学校/大学学习过 DSA 的人来说,所有的例子都感觉纯粹是算法(有充分的理由),但几乎没有任何例子或指导来说明我们每天使用的产品如何利用这个概念。 “我需要这个吗?” 你已经问过很多次这个问题了,不是吗? ? 以下是您今天可以在 React...
    程式設計 發佈於2024-11-03
  • 為什麼表格行上的框陰影在不同瀏覽器中表現不同?
    為什麼表格行上的框陰影在不同瀏覽器中表現不同?
    跨瀏覽器表行上的框陰影外觀不一致應用於表行() 的CSS 框陰影可能表現出不一致的行為跨各種瀏覽器。儘管 CSS 相同,但某些瀏覽器可能會如預期顯示陰影,而其他瀏覽器則可能不會。 要解決此問題,建議將 Transform 屬性與 box-shadow 屬性結合使用。將scale(1,1)加入tran...
    程式設計 發佈於2024-11-03
  • 探索 PHP 中的並發性和並行性:實作教學和技巧
    探索 PHP 中的並發性和並行性:實作教學和技巧
    理解並發性和平行性對於編寫高效的 PHP 應用程式至關重要,特別是在處理需要同時處理的多個任務或操作時。這是理解和實作 PHP 並發性和平行性的逐步指南,包含實作範例和說明。 1.併發與並行 並發:指系統透過交錯執行同時處理多個任務的能力。這並不一定意味著任務是同時執行的,只是對它...
    程式設計 發佈於2024-11-03

免責聲明: 提供的所有資源部分來自互聯網,如果有侵犯您的版權或其他權益,請說明詳細緣由並提供版權或權益證明然後發到郵箱:[email protected] 我們會在第一時間內為您處理。

Copyright© 2022 湘ICP备2022001581号-3