React函式式元件的效能優化

fozero發表於2021-04-24

優化思路

主要優化的方向有2個:

  1. 減少重新 render 的次數。因為在 React 裡最重(花時間最長)的一塊就是 reconction(簡單的可以理解為 diff),如果不 render,就不會 reconction。
  2. 減少計算的量。主要是減少重複計算,對於函式式元件來說,每次 render 都會重新從頭開始執行函式呼叫。

在使用類元件的時候,使用的 React 優化 API 主要是:shouldComponentUpdate和 PureComponent

那麼在函式式元件中,我們怎麼做效能優化?主要用到下面幾個方法去優化

  1. React.memo
  2. useCallback
  3. useMemo

React.memo

看個例子:

我們在父元件中放一個按鈕用於修改子標題,並引入Child子元件

可以看到,第一次進來子元件列印了console.log('我是子元件')

當點選修改子標題,Child子元件也列印了,造成了不必要的重複渲染次數

//父元件
import {useState} from 'react'

import Child from "./Child";
const Index = ()=>{
    const [subTitle, setSubTitle] = useState('我是子標題')
    const updateSubTitle = ()=>{
      setSubTitle('修改子標題')
    }
    return (
      <div>
        <div>函式式元件效能優化</div>
        <div>{subTitle}</div>
        <button onClick={updateSubTitle}>修改子標題</button>
        <Child/>
      </div>
    );
  }
  
  export default Index;


//子元件Child.js
const Child = ()=>{
    console.log('我是子元件')
    return (
        <div>我是子元件</div>
    )
}
export default Child

優化一下,使用React.memo包裹子元件

import React from "react";

const Child = ()=>{
    console.log('我是子元件')
    return (
        <div>我是子元件</div>
    )
}
export default React.memo(Child)

再觀察一下,發現Child子元件沒有重複渲染了

useCallback

這裡我們再改造一下,給Child子元件新增一個onclick事件,然後點選修改子標題按鈕,發現我們的Child子元件又重新渲染了,這裡主要是因為修改子標題的時候handlerClick函式重新渲染變化,造成子元件重新渲染

// 父元件
const Index = ()=>{
    const [subTitle, setSubTitle] = useState('我是子標題')
    const updateSubTitle = ()=>{
      setSubTitle('修改子標題')
    }
    const handlerClick = ()=>{
      console.log('子元件點選')
    }
    return (
      <div>
        <div>函式式元件效能優化</div>
        <div>{subTitle}</div>
        <button onClick={updateSubTitle}>修改子標題</button>
        <Child onClick={handlerClick}/>
      </div>
    );
  }

// Child子元件
const Child = (props)=>{
    console.log('我是子元件')
    return (
        <div>
            <div>我是子元件</div>
            <button onClick={props.onClick}>子元件按鈕</button>
        </div>
    )
}
export default React.memo(Child)

優化一下,使用useCallback包裹處理子元件的handlerClick函式,再次點選updateSubTitle修改子標題,發現Child子元件沒有重新再渲染

// 父元件
const Index = ()=>{
    const [subTitle, setSubTitle] = useState('我是子標題')
    const updateSubTitle = ()=>{
      setSubTitle('修改子標題')
    }
    const handlerClick = useCallback(()=>{
      console.log('子元件點選')
    },[])

    return (
      <div>
        <div>函式式元件效能優化</div>
        <div>{subTitle}</div>
        <button onClick={updateSubTitle}>修改子標題</button>
        <Child onClick={handlerClick}/>
      </div>
    );
  }
  
  export default Index;

這裡關於useCallback的用法

const callback = () => {
  doSomething(a, b);
}

const memoizedCallback = useCallback(callback, [a, b])

把函式以及依賴項作為引數傳入 useCallback,它將返回該回撥函式的 memoized 版本,這個 memoizedCallback 只有在依賴項有變化的時候才會更新。

useMemo

useMemo用於計算結果快取

我們先看個例子,在之前基礎上新增一個calcCount計算函式,然後點選updateSubTitle更新子標題,發現calcCount重新計算了,也就是每次渲染都會造成重複計算,如果是計算量比較大的情況下,會極大的影響效能

// 父元件
const Index = ()=>{
    const [subTitle, setSubTitle] = useState('我是子標題')
    const updateSubTitle = ()=>{
      setSubTitle('修改子標題')
    }
    const handlerClick = useCallback(()=>{
      console.log('子元件點選')
    },[])

    const calcCount = ()=>{
      
      let totalCount = 0
      for(let i=0;i<10000;i++){
        totalCount+=i
      }
      console.log('totalCount',totalCount)
      return totalCount
    }

    const count = calcCount()

    return (
      <div>
        <div>函式式元件效能優化</div>
        <div>{subTitle}</div>
        <button onClick={updateSubTitle}>修改子標題</button>
        <div>count:{count}</div>
        <Child onClick={handlerClick}/>
      </div>
    );
  }

優化一下,使用useMemo快取計算結果,我們再次點選updateSubTitle修改子標題按鈕,可以發現calcCount函式不再重複計算

  const calcCount = ()=>{
      
      let totalCount = 0
      for(let i=0;i<10000;i++){
        totalCount+=i
      }
      console.log('totalCount',totalCount)
      return totalCount
    }

    const count = useMemo(calcCount,[])

最後,需要注意的是不能盲目的使用useMemo,要根據具體的場景,比如對於一個資料計算量比較大,那麼使用是比較適用的,而對於普通的一些值得計算,可以不使用,因為本身useMemo也是會消耗一些效能,盲目使用反而會適得其反

參考閱讀

文章最後

本文作者阿健Kerry,高階前端工程師,轉載請註明出處。如果覺得本文對你有幫助,記得點贊三連哦,也可以掃碼關注我新建立的前端技術公眾號【有你前端】,之後我所有文章會同步發到這個公眾號上面。另外,我建了一個可以幫助我們們程式設計師脫單的公眾號,每週都會推送幾個優秀的單身小姐姐,如果你是程式設計師技術好又正好是單身,那你可以下面掃碼關注【緣來你是程式猿】公眾號開啟你的脫單之旅。

相關文章