正確使用React組件緩存

簡介

正常來講的話當我們點擊組件的時候,該組件以及該組件的子組件都會重新渲染,但是如何避免子組件重新渲染呢,我們經常用memo來解決

React.memo配合useCallback緩存組件

  • 父組件沒有傳props
const Index = ()=> {console.log('子組件刷新了');return (<div>這是子組件</div>)
}
//這里我們用react.memo對組件進行包裹,包裹一次之后react在render的過程中不會給該fiber打上更新的tag
//從而跳過更新,這個原理其實就是react.memo的第二個參數上,如果react.memo第二個參數不傳遞,react回默
//認給我們補充上第二個參數的邏輯,其中邏輯就是淺比較Index組件的props參數,如果相等的話默認第二個參數返
//回true,組件就會緩存了,如果不相等的話就會返回false組件就會重新打上更新的tag然后重新渲染。
const MemoIndex = React.memo(Index);const App = ()=>{const [state, setState] = useState(0);return (<div className="App"><button onClick={()=>setState(state+1)}>點我看看子組件刷新了嗎</button><MemoIndex/></div>);
}
  • 父組件傳了state
const Index = ()=> {console.log('子組件刷新了');return (<div>這是子組件</div>)
}
//其實這Index組件不會更新的,react檢測到我們不傳遞第二個參數的話,會把之前的props拆出來,和現在的
//props做比較 發現pre.next === cur.next 然后回返回true組件就會緩存了
const MemoIndex = React.memo(Index);
//這行代碼就相當這樣的代碼
const MemoIndex = React.memo(Index, (pre, cur)=> {//這樣寫的就比較簡單了,因為這是是針對于當前的demo來說的。react比較的代碼邏輯比較復雜,因為react//需要考慮到多種情況,props中參數可能多一個少一個的情況,所以react默認提供的代碼比較復雜if(pre.name === cur.name) {return true;}return false;
});const App = ()=>{const [state, setState] = useState(0);return (<div className="App"><button onClick={()=>setState(state+1)}>點我看看子組件刷新了嗎</button><MemoIndex name={0}/></div>);
}
  • 父組件傳了函數
const Index = ()=> {console.log('子組件刷新了');return (<div>這是子組件</div>)
}
const MemoIndex = React.memo(Index);const App = ()=>{const [state, setState] = useState(0);const func = ()=> {};return (<div className="App">//這時候子組件會不會刷新呢,有的同學可能說不會因為淺比較發現pre.func === cur.func 返回//true所以不會刷新,但是其實是會刷新的,因為APP組件中觸發了setState之后App組件重新渲染,也//就是相當于執行了App()這個方法,所以里面func的指向地址發生了變化,所以pre.func !== //cur.func  子組件會重新渲染,<button onClick={()=>setState(state+1)}>點我看看子組件刷新了嗎</button><MemoIndex func={func}/></div>);
}

使用useCallback緩存函數

const Index = ()=> {console.log('子組件刷新了');return (<div>這是子組件</div>)
}const MemoIndex = React.memo(Index);const App = ()=>{const [state, setState] = useState(0);//這里我們用了useCallback,useCallback主要是緩存我們當前的函數,如果我們第二個參數傳遞空數組的話//他的地址不會改變,如果我們第二個參數傳遞的是一個變量,這個變量發生變化他的地址就會發生變化。所以這//和useEffect的第二個參數是一樣的,但是請注意不要濫用useCallback的第二個參數。如果第二個參數濫用//會拿到我們之前的值。我們看下一個示例就知道了const func = useCallback(()=> {}, [])return (<div className="App"><button onClick={()=>setState(state+1)}>點我看看子組件刷新了嗎</button><MemoIndex func={func}/></div>);
}

使用useMemo緩存組件

useMemo不僅可以緩存變量,函數還可以緩存組件

const Index = (props)=> {console.log('子組件刷新了');return (<div>這是子組件</div>)
}const App = ()=>{const [state, setState] = useState(0);//使用useMemo也要和useCallback一樣特別注意第二個參數,因為他有可能導致我們拿不到最新的數據解決//解決方案就和useCallback的一樣,簡單來說套用react官方的話就是請確保數組中包含了所有外部作用域//中會隨時間變化并且在useMemo中使用的變量都要放到第二個參數中。const Component = useMemo(()=><Index/>, []);return (<div className="App"><button onClick={()=>setState(state+1)}>點我看看子組件刷新了嗎</button>{Component}</div>);
}

利用props.children緩存組件

這樣在Index組件re-render的時候,由于App(父組件)中的組件沒有變化,所以拿到的children依然是上一次的(沒有發生變化的)所以children部分不會re-render。

const Index = (props)=> {const [state, setState] = useState(0);return (<div>//當這個按鈕點擊之后我們發現Children組件并不重新刷新了,其實原理我理解的是react幫我們做了一層//處理當渲染前與渲染后兩個組件的引用地址一樣他就會放棄render,當然這是我的猜測,這個的話之后//我看到源碼的時候會和大家講一下在補充一下。<button onClick={()=>setState(state+1)}>點我我看看子組件刷新不刷新</button>{props.children}</div>)
};
const Children = ()=> {return (<div>{console.log('子組件刷新了')}這是children組件</div>)
}const App = ()=>{return (<div className="App"><Index><Children/></Index></div>);
}

注意事項

  • useCallBack不是每個函數都需要使用!不要濫用useCallback
const Index = (props)=> {console.log('子組件刷新了');return (<div>//點擊這個按鈕之前先點擊App組件下面的按鈕,讓state變大,然后在點擊這個按鈕看看state是啥//我們發現state一直是一個0。這是為什么呢,因為很簡單我們之前講了useCallback第二個和//useEffect的第二個參數是一樣的,因為我們傳遞的是空數組說以useCallback一直拿到的是最原始的//值,所以會造成這個問題,我們寫代碼的時候千萬要注意第二個參數,只要useCallback需要什//值我們就在第二個參數傳遞什么值,這樣才可以確保我們拿到的是最新的值。同樣的里面如果不需要一些//參數的話我們也不要把這些參數加到第二個參數上面否則會出現func的地址多次改變。<button onClick={props.func}>點我看看state是啥</button>這是子組件</div>)
}
const MemoIndex = React.memo(Index);const App = ()=>{const [state, setState] = useState(0);const func = useCallback(()=> {console.log(state);}, [])return (<div className="App"><button onClick={()=>setState(state+1)}>點我看看子組件刷新了嗎</button><MemoIndex func={func}/></div>);
}
  • useCallBack是一個緩存工具沒錯。但實際上他并不能阻止函數都重現構建

示例:
大家看上方這種結構的組件,Com組件中包含了fun1和fun2兩個函數。

是不是認為當Com組件重新渲染的時候,只有fun2(沒有使用useCallBack的函數)函數會被重新構建,而fun1(使用了useCallBack的函數)函數不會被重新構建。

實際上,被useCallBack包裹了的函數也會被重新構建并當成useCallBack函數的實參傳入。
useCallBack的本質工作不是在依賴不變的情況下阻止函數創建,而是在依賴不變的情況下不返回新的函數地址而返回舊的函數地址。不論是否使用useCallBack都無法阻止組件render時函數的重新創建!!

每一個被useCallBack的函數都將被加入useCallBack內部的管理隊列。而當我們大量使用useCallBack的時候,管理隊列中的函數會非常之多,任何一個使用了useCallBack的組件重新渲染的時候都需要去遍歷useCallBack內部所有被管理的函數找到需要校驗依賴是否改變的函數并進行校驗。

在以上這個過程中,尋找指定函數需要性能,校驗也需要性能。所以,濫用useCallBack不但不能阻止函數重新構建還會增加“尋找指定函數和校驗依賴是否改變”這兩個功能,為項目增添不必要的負擔。

//Com組件
const Com =  () => {//示例1包裹了useCallBack的函數const fun1 = useCallBack(() => {console.log('示例一函數');...},[])//示例2沒有包裹useCallBack的函數const fun2 = () => {console.log('示例二函數');...}return <div></div>
}

不要過度緩存組件

其實不必過度優化代碼 react官方沒有幫你做 其實也證明了 如果你的代碼沒有明顯的卡頓 你自己去做優化 可能造成負優化

優化的手段一般都是針對組件本身比較復雜且數據量大每次re-render都會造成卡頓的情況下才去做的,沒有太明顯的卡頓出現時沒必要做這些優化。而且在使用memo前其實也有手段去規避這些無效的re-render,比如將組件粒度劃分的更細一些

參考文章

useCallBack你真的知道怎么用嗎。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/212744.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/212744.shtml
英文地址,請注明出處:http://en.pswp.cn/news/212744.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Java14道高頻面試題

面試題 1、JWT ①、JWT(全稱:Json Web Token)是一個開放標準(RFC 7519),它定義了一種緊湊的、自包含的方式,用于作為 JSON 對象在各方之間安全地傳輸信息。 ②、JWT 的原理是&#xff0c;服務器認證以后&#xff0c;生成一個 JSON 對象&#xff0c;發回給用戶 ③、JWT是由頭…

機器學習基本概念介紹 2023

筆記來源于&#xff1a; https://www.youtube.com/watch?vphQK8xZpgoU&t172s https://www.youtube.com/watch?vXLyPFnephpY&t645s Machine/Deep Learning 機器學習概況來說&#xff0c;讓機器具備自動找函式的能力 &#xff08;Machine Learning 約等于 Looking …

智能優化算法應用:基于飛蛾撲火算法3D無線傳感器網絡(WSN)覆蓋優化 - 附代碼

智能優化算法應用&#xff1a;基于飛蛾撲火算法3D無線傳感器網絡(WSN)覆蓋優化 - 附代碼 文章目錄 智能優化算法應用&#xff1a;基于飛蛾撲火算法3D無線傳感器網絡(WSN)覆蓋優化 - 附代碼1.無線傳感網絡節點模型2.覆蓋數學模型及分析3.飛蛾撲火算法4.實驗參數設定5.算法結果6.…

訂單系統的設計與海量數據處理實戰

概述 訂單系統可以說是整個電商系統中最重要的一個子系統&#xff0c;因此訂單數據可以算作電商企業最重要的數據資產。訂單系統從代碼上來說可分為兩部分&#xff1a;訂單程序和歷史訂單處理程序。數據存儲進行分庫分表。 訂單系統業務分析 對于一個合格的訂單系統&#xf…

如何使用bash寫腳本

本章主要介紹如何使用bash寫腳本。 了解通配符了解變量了解返回值和數值運算數值的對比判斷語句循環語句 grep的用法是“grep 關鍵字 file”&#xff0c;意思是從file中過濾出含有關鍵字的行。 例如&#xff0c;grep root /var/log/messages&#xff0c;意思是從/var/log/me…

基于Html+騰訊云播SDK開發的m3u8播放器

周末業余時間在家無事&#xff0c;學習了一下騰訊的云播放sdk&#xff0c;并制作了一個小demo&#xff08;m3u8播放器&#xff09;&#xff0c;該在線工具是基于騰訊的云播sdk開發的&#xff0c;云播sdk非常牛&#xff0c;可以支持多種播放格式。 預覽地址 m3u8player.org 源碼…

JVM進程緩存

引言 緩存在日常開發中啟動至關重要的作用&#xff0c;由于是存儲在內存中&#xff0c;數據的讀取速度是非常快的&#xff0c;能大量減少對數據庫的訪問&#xff0c;減少數據庫的壓力。我們把緩存分為兩類&#xff1a; 分布式緩存&#xff0c;例如Redis&#xff1a; 優點&…

Mybatis之簡介、使用操作(安裝、XML、SqlSession、映射的SQL語句、命名空間、作用域和生命周期)

學習的最大理由是想擺脫平庸&#xff0c;早一天就多一份人生的精彩&#xff1b;遲一天就多一天平庸的困擾。各位小伙伴&#xff0c;如果您&#xff1a; 想系統/深入學習某技術知識點… 一個人摸索學習很難堅持&#xff0c;想組團高效學習… 想寫博客但無從下手&#xff0c;急需…

Java項目-瑞吉外賣Day4

實現文件的上傳下載&#xff1a; 前端代碼&#xff1a; 對文件的操作就是對流的操作。 上傳文件的后端代碼&#xff0c;需要注意MultipartFile的名字必須與前端相對&#xff1a; 為文件存儲位置進行動態設置&#xff0c;配置application.xml 在CommonController中設置屬性讀…

Nodejs后端+express框架

前言 基于vue3Node后臺管理項目&#xff0c;補充nodejs和express相關知識。 文章目錄 一&#xff0c;express 1.官網 Express - 基于 Node.js 平臺的 web 應用開發框架 - Express中文文檔 | Express中文網 2.安裝 npm install express --save 二、MongoDB 特點 非關…

uniapp 藍牙小程序

在 uni-app 中開發藍牙相關的小程序涉及到使用 uni-app 提供的藍牙 API。uni-app 為多端開發提供了統一的 API&#xff0c;這意味著你編寫的代碼可以在不同的平臺上運行&#xff0c;包括微信小程序。 以下是實現藍牙功能的基本步驟和代碼示例&#xff1a; 1. 開啟藍牙適配器 …

java之SpringBoot開發實用篇

MENU SpringBoot開發實用篇KF-1.熱部署KF-1-1.手動啟動熱部署KF-1-2.自動啟動熱部署KF-1-3.參與熱部署監控的文件范圍配置KF-1-4.關閉熱部署 KF-2.配置高級KF-2-1.ConfigurationPropertiesKF-2-2.寬松綁定/松散綁定KF-2-3.常用計量單位綁定KF-2-4.校驗KF-2-5.數據類型轉換 KF-3…

【頭歌系統數據庫實驗】實驗8 SQL的復雜多表查詢-2

目錄 第1關&#xff1a;基于派生表查詢每個隊員解答中超過他平均memory的user_id及題目編號problem_id 第2關&#xff1a;用ANY/ALL實現查詢2019級選手&#xff08;user_id前4位為2019&#xff09;滿足比2020級其中一個選手注冊時間早即可的選手 第3關&#xff1a;用聚集查詢…

python zblog API實現類似XMLRPC/發布文章

我發現python對Zblog的XML發布并不友好&#xff0c;雖然也有對應的模塊&#xff0c;但是遠遠沒有XPCRPC更直接方便&#xff0c;但是使用xmlRpc是直接給發布文章帶來了不小的便利&#xff0c;但是對系統也并不友好&#xff0c;但是zblog也開放了Api&#xff0c;但是干部子弟不樂…

UE小:物品拼裝功能

藍圖B1的實現步驟&#xff1a; 獲取玩家控制器和視角&#xff1a;首先獲取玩家控制器&#xff0c;然后使用Deproject Screen to World節點將屏幕上的鼠標位置轉換為世界空間中的一條射線。 射線檢測&#xff1a;使用Line Trace by Channel或Line Trace for Objects節點發射射線…

深度學習測試流程

深度學習模型測試的功能旨在驗證模型在各種情況下的性能和魯棒性。以下是深度學習模型測試的主要功能&#xff1a; 性能評估&#xff1a; 測試模型在任務目標上的整體性能&#xff0c;例如分類準確性、回歸誤差等。評估指標的選擇取決于具體的任務類型。 泛化能力&#xff1a;…

《信息技術時代》期刊雜志論文發表投稿

《信息技術時代》期刊收稿方向&#xff1a;通信工程、大數據、計算機、辦公自動化、信息或計算機教育、電子技術、系統設計、移動信息、圖情信息研究、人工智能、智能技術、信息技術與網絡安全等。 刊名&#xff1a;信息技術時代 主管主辦單位&#xff1a;深圳灣科技發展有限…

C++筆記之int、size_t、uint8_t、unsigned char*區別

C筆記之int、size_t、uint8_t、unsigned char*區別 code review! 文章目錄 C筆記之int、size_t、uint8_t、unsigned char*區別1.ChatGPT第一次查詢解釋2.ChatGPT第二次查詢解釋3.分別的使用示例 1.ChatGPT第一次查詢解釋 size_t、uint8_t 和 int 是編程中使用的不同類型&…

《微信小程序開發從入門到實戰》學習四十七

4.4 云函數 4.4.5 云函數的定時觸發 如果云函數需要定時執行&#xff0c;可以使用云函數定時觸發器。配置了定時觸發器&#xff0c;云函數會在相應時間點被自動觸發。函數返回結果不會返回調用方 在需要添加觸發器的云函數下新建文件config.json。格式如下&#xff1a; &quo…

05-詳解調用服務時負載均衡的配置及其原理

負載均衡 負載均衡的原理(通用) LoadBalanced注解用來攔截它所標記的RestTemplate發起的http請求, 底層是利用了一個名為Ribbon的組件來實現負載均衡功能(Cloud高版本已經棄用) LoadBalancerInterceptor的intercept方法會對RestTemplate的請求進行攔截 public class LoadBal…