+9

How to Optimize React Hooks Performance - Tối ưu hiệu năng khi sử dụng React Hooks

Bài gốc : https://rahmanfadhil.com/optimize-react-hooks/


Kể từ version 16.8, React giới thiệu một feature mới gọi là React Hooks. Một cách ngắn gọn nhất, React Hook là phiên bản nâng cấp của function component, cho phép function component sử dụng các tính năng mà trước đó chỉ class component mới có như life cycle hooks, state.

Nhưng như trong phim kiếm hiệp hay có câu, thanh gươm càng có nhiều sức mạnh thì càng khó điều khiển, chúng ta hãy cùng nhau tìm hiểu xem, làm thể nào để điều khiển được thanh gươm React Hooks này một cách hiệu quả nhất.

Getting started

Đầu tiên, tạo một project mới hoàn toàn sử dụng Create React App

$ npx create-react-app my-app

Sau đó, viết lại toàn bộ file App.js như sau:

// ./src/App.js
import React, { useState } from "react"
import Counter from "./Counter"

export default function App() {
  const [value, setValue] = useState("")

  return (
    <div>
      <input
        type="text"
        onChange={(e) => setValue(e.target.value)}
        value={value}
      />
      <Counter />
    </div>
  )
}

Trong App component, chúng ta có một thẻ input, với giá trị được điều khiển bởi biến value được tạo ra bởi useState hook. Đồng thời, chúng ta cũng cho hiển thị Counter component được import từ ./Counter.js, giờ thì hãy tạo ra Counter component:

// ./src/Counter.js
import React, { useState, useRef } from "react"

export default function Counter() {
  const [counter, setCounter] = useState(0)
  const renders = useRef(0)

  return (
    <div>
      <div>Counter: {counter}</div>
      <div>Renders: {renders.current++}</div>
      <button onClick={() => setCounter(counter + 1)}>Increase Counter</button>
    </div>
  )
}

Trong Counter component này, chúng ta sẽ có một counter state, nó sẽ tăng lên mỗi lần chúng ta nhấn nút Increase Counter, và một renders ref để "keep track" số lần component của chúng ta được re-render, hay nói cách khác là cái đống bên trong return được chạy lại.

Nói thêm về việc sử dụng useRef ở đây, sẽ có nhiều bạn cũng như mình khi lần đầu nhìn thấy cái useRef này, chắc chắn sẽ thắc mắc và tự hỏi tại sao lại dùng useRef. Thì câu trả lời là do chúng ta muốn tính số lần re-render của component, và useRef cho chúng ta chính xác cái chúng ta đang cần. Vì khi thay đổi giá trị của .current của ref, nó sẽ không trigger việc re-render. Các bạn có thể đọc thêm về đặc điểm này của useRef tại useRefIs there something like instance variables?

The unnecessary rerender

Bây giờ, các bạn thử yarn start để chạy ứng dụng lên, ta sẽ thấy mỗi lần nhấn nút "Increase Counter", con số counter sẽ đúng bằng số lần render. Điều này có nghĩa là Counter component sẽ được re-render (hay nói cách khác là cái đống jsx trong return được chạy lại) mỗi khi state thay đổi.

(NOTE: Nhớ bỏ đi React.StrictMode ở file index.js, vì nó sẽ khiến component render twice)

Nhưng, khi ta gõ vào ô input của App component, ta thấy con số renders vẫn tăng lên. Điều này có nghĩa là Counter component cũng sẽ re-render mỗi khi cái state của ô input ở App componnet thay đổi, điều này là không cần thiết vì chẳng có gì ta muốn thay đổi bên trong Counter component.

Chạy lại đống code trong Counter component là không cần thiết, vậy thì làm sao để ta có thể tránh việc này ?

Memoizing components

Từ phiên bản 16.6, ta có thêm React.memo có tính năng tương tự PureComponent hay shouldComponentUpdate bên class component, nhưng React.memo là cho function component. Mấy thứ fancy stuffs này mục đích sử dụng chủ yếu để optimize performance, nghĩa là nó sẽ giúp ta giảm số lần render. Nghĩa là nó chỉ render khi mà input prop có thay đổi, còn nếu giống thì nó sẽ tự "bail out". Nói tới đây, thì các bạn có thể nghiên cứu thêm về cơ chế mà React quyết định như nào là giống, như nào là khác, nghĩa là thuật toán nó dùng để so sánh props để xác định có sự thay đổi.

Bây giờ chúng ta hãy thử dùng React.memo để bọc cái Counter của chúng ta lại.

// ./src/Counter.js
import React, { useState, useRef } from "react"

export default React.memo(() => {
  const [counter, setCounter] = useState(0)
  const renders = useRef(0)

  return (
    <div>
      <div>Counter: {counter}</div>
      <div>Renders: {renders.current++}</div>
      <button onClick={() => setCounter(counter + 1)}>Increase Counter</button>
    </div>
  )
})

Đơn giản, dễ xài! Giờ thử mở app của chúng ta lên, và ta sẽ thấy Counter component sẽ không re-render mỗi khi ta nhập vào ô input.

Khi ta truyền một prop vào memoried component, thì memoried component này sẽ kiểm tra xem là prop có thay đổi hay không để quyết định có re-render component hay không. Để kiểm tra tính năng này, hãy thử truyền một prop vào Counter component.

// ./src/App.js
import React, { useState } from "react"
import Counter from "./Counter"

export default function App() {
  const [value, setValue] = useState("")

  return (
    <div>
      <input
        type="text"
        onChange={(e) => setValue(e.target.value)}
        value={value}
      />
      <Counter greeting="Hello world!" />
    </div>
  )
}

Ở đây, chúng ta đang truyền greeting prop vào Counter component. Nếu bạn chạy lại app, nó sẽ vẫn như khi vừa nãy. Bởi vì memoried component sẽ chỉ cập nhật lại khi prop thay đổi.

Memoizing functions

React.memo is awesome, nhưng nó cũng có nhược điểm. Nó sẽ hoạt động tốt với các kiểu dữ liệu như string, number, boolean. Còn với objects và functions, nó sẽ không đủ thông minh để kiểm tra chính xác sự thay đổi.

// ./src/App.js
import React, { useState } from "react"
import Counter from "./Counter"

export default function App() {
  const [value, setValue] = useState("")

  return (
    <div>
      <input
        type="text"
        onChange={(e) => setValue(e.target.value)}
        value={value}
      />
      <Counter
        addHello={() => setValue(value + "Hello!")}
        myObject={{ key: "value" }}
      />
    </div>
  )
}

Ở đây ta đang truyền cho Counter hai prop mới, một function và một object. Và lúc này, mỗi khi ta nhập vô ô input của App component, thì Counter component của chúng ta lại re-render. Điều này đồng nghĩa, với mỗi lần chạy, props đi vào Counter component là một function và một object khác nhau. Props thay đổi nên component re-render.

Để giải quyết vấn đề này, chúng ta có thể sử dụng useCallback, để memoize cái function trước khi pass xuống cho Counter component. Cái memozied version của cái function sẽ chỉ thay đổi khi mà một trong các dependencies thay đổi. Hãy sửa code trong App.js lại như sau:

// ./src/App.js
import React, { useState, useCallback } from "react"
import Counter from "./Counter"

export default function App() {
  const [value, setValue] = useState("")

  const addHello = useCallback(() => setValue(value + "Hello!"), [value])

  return (
    <div>
      <input
        type="text"
        onChange={(e) => setValue(e.target.value)}
        value={value}
      />
      <Counter addHello={addHello} myObject={{ key: "value" }} />  {/* khi chạy app, hãy tạm xóa cái myObject nhé  */}
    </div>
  )
}

Phương pháp này sẽ rất hữu ích khi chúng ta có nhiều hơn một state hook. Cái memoized function sẽ chỉ được chạy lại khi mà cái chosen state bị thay đổi. Để minh họa cho điều này, hãy thêm vào một ô input nữa:

import React, { useState, useCallback } from "react"
import Counter from "./Counter"

export default function App() {
  const [value, setValue] = useState("")
  const [newValue, setNewValue] = useState("")

  const addHello = useCallback(() => setValue(value + "Hello!"), [value])

  return (
    <div>
      <input
        type="text"
        onChange={(e) => setValue(e.target.value)}
        value={value}
      />
      <input
        type="text"
        onChange={(e) => setNewValue(e.target.value)}
        value={newValue}
      />
      <Counter addHello={addHello} myObject={{ key: "value" }} /> {/* khi chạy app, hãy tạm xóa cái myObject nhé */}
    </div>
  )
}

Bây giờ, khi ta nhấn vô ô input mới, cái Counter component sẽ không re-render, bởi vì chúng ta đã chỉ ra trong cái [] dependencies là nó chỉ phụ thuộc vào gia trị value, nó chỉ được chạy lại khi mà cái giá trị đó thay đổi, còn những giá trị khác thay đổi thì mặc kệ.

Memoizing objects

Tới đây, chúng ta đã biết cách memoize các function, nhưng còn một thứ nữa chúng ta nên biết về momoizing.

Hiện tại, cái Counter component của chúng ta vẫn bị re-render mỗi khi state thay đổi. Đó là bởi vì cái myObject prop vẫn chưa được memoized. Chúng ta có thể sử dụng useMemo để memoize tất cả các giá trị (bao gồm cả objects) bằng cách truyền vào một "create" function và một mảng các dependencies. Cái giá trị được tính toán ra sẽ chỉ được tính toán lại khi mà một trong các dependencies thay đổi (giống hệt như useCallback ở trên).

import React, { useState, useCallback, useMemo } from "react"
import Counter from "./Counter"

export default function App() {
  const [value, setValue] = useState("")
  const [newValue, setNewValue] = useState("")

  const addHello = useCallback(() => setValue(value + "Hello!"), [value])
  const myObject = useMemo(() => ({ key: "value" }), [])

  return (
    <div>
      <input
        type="text"
        onChange={(e) => setValue(e.target.value)}
        value={value}
      />
      <input
        type="text"
        onChange={(e) => setNewValue(e.target.value)}
        value={newValue}
      />
      <Counter addHello={addHello} myObject={myObject} />
    </div>
  )
}

Conclusion

Bằng cách áp dụng các phương pháp này, chúng ta sẽ có thể passs props vào một memozied component một cách tối ưu nhất.


All rights reserved

Viblo
Hãy đăng ký một tài khoản Viblo để nhận được nhiều bài viết thú vị hơn.
Đăng kí