React.memo 和 useMemo 的使用問題小結
問題背景
大家在使用 React 框架進行開發(fā)時一定遇到過以下問題:
- 當函數(shù)式組件中的某一狀態(tài)改變,整個組件刷新,重新渲染
- 在類組件中 setState() 時,整個組件也會重新渲染
- 以上問題若不進行優(yōu)化,導致的結果是:
- 隨著代碼的增加,每次的狀態(tài)改變,頁面進行一次 reRender ,這將產(chǎn)生很多不必要的 reRender 不僅浪費性能,從而導致頁面卡頓;
useMemo 進行優(yōu)化
以下面 App 組件進行分析
import './App.css';
import ProfileTest from './components';
import { Profiler, useEffect, useMemo, useState, useRef } from 'react'
function App () {
const [name, setName] = useState('')
const [num, setNum] = useState(0)
useEffect(() => {
setTimeout(() => {
console.log('111')
setName('xxx')
}, 2000)
}, [])
const memoVal = useMemo(() => {
console.log('運行了useMemo num值為:', num);
return num + 1
}, [num])
console.log('memoVal值為:', memoVal)
console.log('父組件運行分割線----------------------------------------------------')
const changeNum = () => {
setNum(2)
}
return (
<Profiler id='profile-test'>
<div className="App">
{/* <ProfileTest /> */}
<button style={{ marginTop: 100 }} onClick={changeNum}>改變num</button>
</div>
</Profiler>
);
}
export default App;以上組件在首次渲染、以及 2秒后的執(zhí)行結構如下圖所示:

很顯然首次渲染執(zhí)行了,useMemo,而2秒后有狀態(tài)變化后沒有執(zhí)行useMemo。
點擊按鈕改變 useMemo 的依賴項后可以發(fā)現(xiàn),如下圖所示 useMemo 又執(zhí)行了。

因此在使用函數(shù)式組件時,可以使用 useMemo 減少不必要的reRender 提高組件的性能;
React.memo 進行優(yōu)化
在以上組件的基礎上,給App 增加一個子組件,代碼如下所示:
import React from 'react'
export default function Children(props) {
console.log('子組件運行了,接收的props是', props)
console.log('子組件渲染分割線------------------------------------------')
return <div>子組件</div>
}首次render 以及 2s后組件的 reRender 控制臺打印結果如下圖所示:

由上圖可以看出,reRender 時Children 組件的props并未變化,因此,此次Children 組件的reRender 是不必要的,需要進行優(yōu)化;
props的值是基本類型
如果 Children 的 props 是基本類型,則可以做一下優(yōu)化:
import React, { memo } from 'react'
function Children(props) {
console.log('子組件運行了,接收的props是', props)
console.log('子組件渲染分割線------------------------------------------')
return <div>子組件</div>
}
export default memo(Children)優(yōu)化后控制臺打印如下信息,一下信息可以看出 Children 組件沒有進行 reRender

props的值是引用類型
若子組件的 props 是引用類型 ,則需要進行深度比較,此時React.memo()要傳入第二個參數(shù)進行深度比較,改變后 Children 組件的代碼如下所示:
import React, { memo } from 'react'
function Children(props) {
console.log('子組件運行了,接收的props是', props)
console.log('子組件渲染分割線------------------------------------------')
return <div>子組件</div>
}
export default memo(Children, (preProps, nextProps) => {
return JSON.stringify(preProps) === JSON.stringify(nextProps)
})以上 memo 第二個參數(shù) ,通過比較 preProps 和 nextProps 返回一個布爾值,使得props 進行深度比較;注意:React.memo的第二個參數(shù)進行深度比較時有一定開銷,其產(chǎn)生的開銷存在大于子組件reRender的可能
寫在最后
useMemo() 和 React.memo() 都是進行組件性能優(yōu)化的方式,其區(qū)別是
- useMemo 可以進行更加細粒度的優(yōu)化(有依賴項)
- React.memo() 可以控制props的淺比較和深度比較
- React.memo在沒有第二個參數(shù)的時候相當于class中的PureComponent,當增加了第二個參數(shù)的時候相當于生命周期中的shouldComponentUpdate;
到此這篇關于React.memo 和 useMemo 的使用問題小結的文章就介紹到這了,更多相關React.memo 和 useMemo內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
react搭建環(huán)境時執(zhí)行npm start報錯start: 'react-scripts&
這篇文章主要介紹了react搭建環(huán)境時執(zhí)行npm start報錯start: 'react-scripts start'的解決方案,具有很好的參考價值,希望杜對大家有所幫助,2023-10-10
react?路由權限動態(tài)菜單方案配置react-router-auth-plus
這篇文章主要為大家介紹了react路由權限動態(tài)菜單方案react-router-auth-plus傻瓜式配置詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-08-08

