+3

Mọi thứ cần biết về Re-Render

Bạn có từng ngồi trước màn hình, gõ vào một ô nhập input, và nhận ra rằng mỗi ký tự xuất hiện với một độ trễ khó chịu không? 1_D1D_LeQm6o_6prrIi2Jw3w (1).gif

Tôi đoán câu trả lời là có. Và tôi cũng đoán rằng phản ứng đầu tiên của bạn là:

"Code của tôi không có vấn đề gì mà, đọc vẫn oke mà”

Code của bạn có thể hoạt động hoàn hảo về mặt logic, nhưng khi ứng dụng lớn lên, nó sẽ càng ngày càng chậm, vậy thủ phạm chính là gì?

React re-renders - một trong những khái niệm vừa đơn giản vừa phức tạp nhất mà mọi React developer cần hiểu cực kỳ rõ


Nếu thấy hay thì cho mình xin một bookmark ở phía bên trái để bài viết đến được nhiều người hơn nha

Chúng mình có tạo Group cho các bạn cùng chia sẻ và học hỏi về frontend nâng cao, các câu phỏng vấn khó nha 😄😄😄

Các bạn tham gia để gây dựng cộng đồng Frontend Việt Nam thật lớn mạnh nhé 😍😍😍

Cộng Đồng Frontend Nâng Cao Việt Nam: https://www.facebook.com/groups/seniorfrontendvietnam

Kênh TikTok: https://www.tiktok.com/@sydexa.com

Kênh youtube: https://www.youtube.com/@sydexa.official

Nếu thấy hay thì cho mình xin một bookmark ở phía bên trái để bài viết đến được nhiều người hơn nha


Tại sao bạn nên đào sâu về re-renders?

Để trả lời câu hỏi này, hãy tưởng tượng React như một người họa sĩ đang vẽ lại căn phòng của bạn. Mỗi lần có thay đổi - dù chỉ là di chuyển một cái ghế - họa sĩ có thể:

  • Thông minh: Chỉ vẽ lại cái ghế đó (re-render cần thiết)

  • Hơi không thông minh cho lắm: Vẽ lại toàn bộ căn phòng (re-render không cần thiết)

image.png

React rất nhanh, nhưng nếu bạn bắt nó "vẽ lại toàn bộ căn phòng" sau mỗi thay đổi nhỏ, cuối cùng người dùng sẽ thấy ứng dụng của bạn lag. Và một khi họ cảm thấy lag, trải nghiệm đã bị phá hỏng.

💡Không phải tất cả re-renders đều xấu!

Re-render cần thiết

Có những re-render được coi là cần thiết, vì component cần cập nhật vì dữ liệu của chính nó thay đổi.

image.png

Ví dụ: Bạn gõ vào một ô input. Component quản lý state của input đó phải re-render sau mỗi ký tự để hiển thị những gì bạn vừa gõ. Đây là điều hoàn toàn bình thường và cần thiết.

Re-render không cần thiết

Đây là re-render không cần thiết - component re-render mặc dù không có dữ liệu nào liên quan đến nó thay đổi.

Ví dụ: Bạn gõ vào một input field, nhưng toàn bộ trang web - bao gồm header, footer, sidebar - đều re-render theo. Đây là lãng phí.

image.png

React được tối ưu cực kỳ tốt. Hầu hết thời gian, những re-renders không cần thiết này này xảy ra quá nhanh đến nỗi người dùng không nhận ra. NHƯNG Nếu re-renders xảy ra quá thường xuyên hay trên các component quá "nặng”. Ứng dụng sẽ bắt đầu lag, phản hồi chậm, và dự án mất dần khách hàng

Dưới đây là 5 nguyên nhân phổ biến nhất chính khiến một component re-render, hãy nhớ thuộc lòng hoặc lưu blog này lại nha

1. State thay đổi - Nguồn gốc của mọi re-render

Đây là nguyên nhân phổ biến và dễ hiểu nhất

image.png

Khi bạn gọi hàm thay đổi giá trị count, component chứa nó sẽ bị rerender

Và đây là nguồn gốc - của mọi re-render. Tất cả những gì xảy ra sau đây đều là hệ quả.

2. Parent Re-renders - Hiệu ứng domino

Đây là nơi mọi thứ trở nên thú vị (và đôi khi nó không vui cho lắm =)) )

💡 Hãy nhớ kỹ: Khi cha bị re-render, tất cả con của nó cũng re-render.

image.png

Re-render luôn đi "xuống" cây component (top-down). Re-render của con không bao giờ trigger re-render của cha trên cây

3. Context thay đổi - Sức mạnh đi kèm trách nhiệm

Context là một công cụ mạnh mẽ, nhưng cũng có thể là con dao hai lưỡi:

image.png

Tất cả components sử dụng Context sẽ re-render khi value thay đổi, ngay cả khi component chỉ dùng một phần của data

4. Khi Hooks thay đổi

Bất kể bạn có bao nhiêu custom hooks, tất cả đều "thuộc về" component sử dụng chúng. State hay Context thay đổi trong bất kỳ hook nào cũng sẽ khiến re-render của component chứa chúng.

image.png

5. Key Prop thay đổi

Đây là một trường hợp đặc biệt. key là một prop đặc biệt của React, không phải là prop thông thường, nó giúp React theo dõi component chứ không KHÔNG truyền vào component

image.png

Khi key prop thay đổi, React không re-render component mà thay vào đó, nó unmount component cũ và mount một component hoàn toàn mới. Khi key thay đổi, React hiểu rằng: "Đây là một component instance hoàn toàn khác nên loại bỏ cái cũ luôn".

image.png

Và điều nguy hiểm ở đây là: Unmount/remount tốn tài nguyên hơn hơn re-render bình thường. Component sẽ mất toàn bộ state, phải chạy lại tất cả logic khởi tạo, chạy lại useEffect cleanup và setup mới

Nên là trong list thì đừng dùng index làm key như này nha, sydexa đã có một video nói riêng về phần này đó

image.png

Và cuối cùng có phải là Props thay đổi gây Re-render không?

Đây là một trong những quan niệm sai lầm phổ biến nhất về React

Props thay đổi KHÔNG trực tiếp gây re-render bạn nhé

Hãy xem lại ví dụ sau:

image.png

Để count prop của Child thay đổi, Parent phải re-render trước (vì Parent quản lý state mà). Và khi Parent re-render, Child tự động re-render theo (theo quy tắc 2 mà mình vừa nói ở trên nha)

Props chỉ trở nên quan trọng khi bạn sử dụng memoization techniques như React.memo() , useMemo(), useCallback(). Khi đó, React mới so sánh props cũ và mới để quyết định có cần re-render hay không.

Re-renders không phải là kẻ thù. Chúng là cách React cập nhật UI để phản ánh state mới. Vấn đề chỉ nằm ở những re-renders không cần thiết xảy ra quá nhiều hoặc trên components quá nặng

Hãy thuộc nằm lòng 5 quy tắc trên để cải thiện chất lượng dự án của anh em nha


Nếu thấy hay thì cho mình xin một bookmark ở phía bên trái để bài viết đến được nhiều người hơn nha

Chúng mình có tạo Group cho các bạn cùng chia sẻ và học hỏi về frontend nâng cao, các câu phỏng vấn khó nha 😄😄😄

Các bạn tham gia để gây dựng cộng đồng Frontend Việt Nam thật lớn mạnh nhé 😍😍😍

Cộng Đồng Frontend Nâng Cao Việt Nam: https://www.facebook.com/groups/seniorfrontendvietnam

Kênh TikTok: https://www.tiktok.com/@sydexa.com

Kênh youtube: https://www.youtube.com/@sydexa.official

Nếu thấy hay thì cho mình xin một bookmark ở phía bên trái để bài viết đến được nhiều người hơn nha



All Rights Reserved

Viblo
Let's register a Viblo Account to get more interesting posts.