」工欲善其事,必先利其器。「—孔子《論語.錄靈公》
首頁 > 程式設計 > 掌握 React 重新渲染:如何避免常見陷阱

掌握 React 重新渲染:如何避免常見陷阱

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

Mastering React Re-renders: How to Avoid Common Pitfalls

1.介绍

想象一下:您刚刚构建了一个时尚、功能丰富的 React 应用程序,但突然间,性能开始下降。按钮感觉迟钝,数据更新时间太长,而且你不知道为什么。如果这听起来很熟悉,那么您并不孤单。如果处理不当,React 重新渲染通常是性能问题背后的无声罪魁祸首。

掌握 React 中的重新渲染对于旨在构建高效、可扩展应用程序的 Web 开发人员至关重要。让我们深入研究 React 重新渲染的工作原理,并发现优化性能、避免常见陷阱并将您的 React 应用程序转变为极快体验的实用策略。

2. React 重新渲染如何工作?

React 的重新渲染机制是其最强大的功能之一。它允许您的应用程序根据状态或属性的变化动态更新 UI。但是,如果不进行优化,此过程可能会导致不必要的重新渲染,从而导致性能下降。

简单来说,React 每次都会重新渲染:

  1. 组件的状态发生变化。
  2. 组件接收新的 props。
  3. 父组件重新渲染。

但是当组件重新渲染过于频繁时会发生什么?浏览器必须重新绘制屏幕,​​从而导致动画卡顿、交互缓慢和用户沮丧。现在,问问自己:有多少次您的应用程序感觉速度很慢,而您却无法确切地指出原因?

3.为什么过度重新渲染会损害性能

过多的重新渲染会产生多米诺骨牌效应,使您的整个应用程序变得缓慢。每次重新渲染发生时:

  1. React 使更改与虚拟 DOM 协调。
  2. 它计算对实际 DOM 进行的最少更新。
  3. 浏览器重新绘制屏幕。

在小型应用程序中,这可能不明显。但在具有深度嵌套组件和复杂状态管理的大型应用程序中,即使是最小的不必要的重新渲染也可能会形成性能瓶颈。

这是一个简单的例子:

const UserList = ({ users }) => {
  return users.map(user => 
{user.name}
); }; // Inside a parent component:

现在想象一下,每次父组件更新时,这个 UserList 都会重新呈现,即使用户没有更改。这就是问题所在。我们如何防止这种不必要的重新渲染?

4.常见陷阱(以及如何避免它们)

陷阱 1:不记忆函数和对象

每次父组件重新渲染时,它都会创建新的函数和对象引用,即使实际值没有改变。这会导致子组件不必要地重新渲染。

操作:使用 useCallback 和 useMemo 挂钩来记忆函数和对象。

const ParentComponent = () => {
  const handleClick = useCallback(() => {
    // some function logic
  }, []);

  const userDetails = useMemo(() => ({ name: "Rajesh", age: 30 }), []);

  return ;
};

为什么有效: 记忆化可以防止在每次重新渲染时创建新的函数和对象引用,从而减少子组件中不必要的渲染。

陷阱 2:将内联函数或对象作为 Props 传递

每次组件渲染时都会创建内联函数或对象,导致接收组件重新渲染,即使实际值没有更改。

操作:在组件外部声明函数和对象或使用记忆挂钩。

// Avoid this:
 console.log("clicked")} />

// Instead:
const handleClick = useCallback(() => console.log("clicked"), []);

陷阱3:未能使用React.memo

一些组件本质上是纯粹的——它们总是在给定相同的 props 的情况下渲染相同的输出。然而,如果没有 React.memo,当父组件重新渲染时,它们仍然会重新渲染。

操作:用 React.memo 包裹功能组件,以防止不必要的重新渲染。

const ChildComponent = React.memo(({ data }) => {
  return 
{data.name}
; });

为什么有效: React.memo 确保组件仅在其 props 更改时重新渲染,避免冗余渲染。

5.如何跟踪和调试 React 重新渲染

有没有想过,“为什么我的组件会重新渲染?” React DevTools 提供了一种很好的方法来跟踪渲染并了解哪里出了问题。

  1. 启用“突出显示更新”:这会显示哪些组件正在重新渲染。
  2. 使用“Profiler”选项卡:它显示渲染计时并帮助识别瓶颈。

您还可以使用像这样的自定义挂钩来记录重新渲染:

const useWhyDidYouRender = (componentName, props) => {
  const previousProps = useRef(props);

  useEffect(() => {
    if (previousProps.current !== props) {
      console.log(`${componentName} re-rendered`);
      previousProps.current = props;
    }
  });
};

6。优化 React 重新渲染的最佳实践

1.在需要时提升状态,但并非总是如此

有时,将状态提升到更高的组件会导致子组件不必要的重新渲染。相反,尽可能在本地管理状态。

2.明智地使用密钥

渲染列表时,确保每个项目都有一个稳定、唯一的 key prop。这有助于 React 通过识别哪些项目已更改来优化重新渲染。

3.避免重新渲染未安装的组件

这可能会导致内存泄漏和性能问题。使用 useEffect 清理函数来避免在未挂载的组件中重新渲染。

useEffect(() => {
  return () => {
    // cleanup logic here
  };
}, []);

7.总结:掌握 React 重新渲染

简而言之,React 重新渲染可能是性能问题的隐藏根源,但通过正确的技术,您可以防止不必要的重新渲染,保持应用程序快速运行,并确保流畅的用户体验。

  • 记忆函数和对象以避免重新渲染子组件。
  • 使用 React.memo 停止纯组件中不必要的重新渲染。
  • 利用 React DevTools 跟踪、了解和调试性能瓶颈。

8.实际应用:通过有意的重新渲染提高性能

下次你优化 React 应用程序时,请批判性地思考每次重新渲染。它们有必要吗?如果你记住这个函数会发生什么?通过应用这些策略,您将编写更精简、更快的 React 代码,并避免即使是最有经验的开发人员也会遇到的常见陷阱。


最终想法: 既然您已经了解了 React 重新渲染的细节,请继续将这些优化应用到您自己的应用程序中。您不仅会看到更好的性能,还会获得编写可扩展、可维护的 React 代码的信心!


版本聲明 本文轉載於:https://dev.to/paharihacker/mastering-react-re-renders-how-to-avoid-common-pitfalls-k01?1如有侵犯,請聯絡[email protected]刪除
最新教學 更多>
  • 編譯器報錯“usr/bin/ld: cannot find -l”解決方法
    編譯器報錯“usr/bin/ld: cannot find -l”解決方法
    錯誤:“ usr/bin/ld:找不到-l “ 此錯誤表明鏈接器在鏈接您的可執行文件時無法找到指定的庫。為了解決此問題,我們將深入研究如何指定庫路徑並將鏈接引導到正確位置的詳細信息。 添加庫搜索路徑的一個可能的原因是,此錯誤是您的makefile中缺少庫搜索路徑。要解決它,您可以在鏈接器命令中添...
    程式設計 發佈於2025-05-22
  • 如何使用組在MySQL中旋轉數據?
    如何使用組在MySQL中旋轉數據?
    在關係數據庫中使用mySQL組使用mySQL組進行查詢結果,在關係數據庫中使用MySQL組,轉移數據的數據是指重新排列的行和列的重排以增強數據可視化。在這裡,我們面對一個共同的挑戰:使用組的組將數據從基於行的基於列的轉換為基於列。 Let's consider the following ...
    程式設計 發佈於2025-05-22
  • 如何使用Python的請求和假用戶代理繞過網站塊?
    如何使用Python的請求和假用戶代理繞過網站塊?
    如何使用Python的請求模擬瀏覽器行為,以及偽造的用戶代理提供了一個用戶 - 代理標頭一個有效方法是提供有效的用戶式header,以提供有效的用戶 - 設置,該標題可以通過browser和Acterner Systems the equestersystermery和操作系統。通過模仿像Chro...
    程式設計 發佈於2025-05-22
  • eval()vs. ast.literal_eval():對於用戶輸入,哪個Python函數更安全?
    eval()vs. ast.literal_eval():對於用戶輸入,哪個Python函數更安全?
    稱量()和ast.literal_eval()中的Python Security 在使用用戶輸入時,必須優先確保安全性。強大的Python功能Eval()通常是作為潛在解決方案而出現的,但擔心其潛在風險。 This article delves into the differences betwee...
    程式設計 發佈於2025-05-22
  • 如何在Java字符串中有效替換多個子字符串?
    如何在Java字符串中有效替換多個子字符串?
    在java 中有效地替換多個substring,需要在需要替換一個字符串中的多個substring的情況下,很容易求助於重複應用字符串的刺激力量。但是,對於大字符串或使用許多字符串時,這可能是降低的。 利用正則表達式Example UsageConsider a scenario where ...
    程式設計 發佈於2025-05-22
  • `console.log`顯示修改後對象值異常的原因
    `console.log`顯示修改後對象值異常的原因
    foo = [{id:1},{id:2},{id:3},{id:4},{id:id:5},],]; console.log('foo1',foo,foo.length); foo.splice(2,1); console.log('foo2', foo, foo....
    程式設計 發佈於2025-05-22
  • 如何使用Depimal.parse()中的指數表示法中的數字?
    如何使用Depimal.parse()中的指數表示法中的數字?
    在嘗試使用Decimal.parse(“ 1.2345e-02”中的指數符號表示法表示的字符串時,您可能會遇到錯誤。這是因為默認解析方法無法識別指數符號。 成功解析這樣的字符串,您需要明確指定它代表浮點數。您可以使用numbersTyles.Float樣式進行此操作,如下所示:[&& && && ...
    程式設計 發佈於2025-05-22
  • 如何解決由於Android的內容安全策略而拒絕加載腳本... \”錯誤?
    如何解決由於Android的內容安全策略而拒絕加載腳本... \”錯誤?
    Unveiling the Mystery: Content Security Policy Directive ErrorsEncountering the enigmatic error "Refused to load the script..." when deployi...
    程式設計 發佈於2025-05-22
  • 為什麼HTML無法打印頁碼及解決方案
    為什麼HTML無法打印頁碼及解決方案
    無法在html頁面上打印頁碼? @page規則在@Media內部和外部都無濟於事。 HTML:Customization:@page { margin: 10%; @top-center { font-family: sans-serif; font-weight: ...
    程式設計 發佈於2025-05-22
  • \“(1)vs.(;;):編譯器優化是否消除了性能差異?\”
    \“(1)vs.(;;):編譯器優化是否消除了性能差異?\”
    答案: 在大多數現代編譯器中,while(1)和(1)和(;;)之間沒有性能差異。編譯器: perl: 1 輸入 - > 2 2 NextState(Main 2 -E:1)V-> 3 9 Leaveloop VK/2-> A 3 toterloop(next-> 8 last-> 9 ...
    程式設計 發佈於2025-05-22
  • 如何從Python中的字符串中刪除表情符號:固定常見錯誤的初學者指南?
    如何從Python中的字符串中刪除表情符號:固定常見錯誤的初學者指南?
    從python import codecs import codecs import codecs 導入 text = codecs.decode('這狗\ u0001f602'.encode('utf-8'),'utf-8') 印刷(文字)#帶有...
    程式設計 發佈於2025-05-22
  • 您如何在Laravel Blade模板中定義變量?
    您如何在Laravel Blade模板中定義變量?
    在Laravel Blade模板中使用Elegance 在blade模板中如何分配變量對於存儲以後使用的數據至關重要。在使用“ {{}}”分配變量的同時,它可能並不總是最優雅的解決方案。 幸運的是,Blade通過@php Directive提供了更優雅的方法: $ old_section =...
    程式設計 發佈於2025-05-22
  • Python元類工作原理及類創建與定制
    Python元類工作原理及類創建與定制
    python中的metaclasses是什麼? Metaclasses負責在Python中創建類對象。就像類創建實例一樣,元類也創建類。他們提供了對類創建過程的控制層,允許自定義類行為和屬性。 在Python中理解類作為對象的概念,類是描述用於創建新實例或對象的藍圖的對象。這意味著類本身是使用...
    程式設計 發佈於2025-05-22
  • 如何在GO編譯器中自定義編譯優化?
    如何在GO編譯器中自定義編譯優化?
    在GO編譯器中自定義編譯優化 GO中的默認編譯過程遵循特定的優化策略。 However, users may need to adjust these optimizations for specific requirements.Optimization Control in Go Compi...
    程式設計 發佈於2025-05-22
  • Go語言垃圾回收如何處理切片內存?
    Go語言垃圾回收如何處理切片內存?
    在Go Slices中的垃圾收集:詳細的分析在GO中,Slice是一個動態數組,引用了基礎陣列。使用切片時,了解垃圾收集行為至關重要,以避免潛在的內存洩漏。 考慮使用slice使用slice的以下實現:字符串{ R:=(*Q)[0] *q =(*q)[1:len(*q)] 返...
    程式設計 發佈於2025-05-22

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

Copyright© 2022 湘ICP备2022001581号-3