欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

react組件多次渲染問題的解決

 更新時(shí)間:2023年07月20日 10:21:44   作者:Ruonorth  
本文主要介紹了react組件多次渲染問題的解決,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

問題背景

在數(shù)據(jù)沒有發(fā)生變化的情況下React組件會(huì)進(jìn)行數(shù)次重復(fù)渲染,繪制出來完全相同的兩個(gè)圖

排查思路

尋找子組件重渲染原因?qū)嶒?yàn)

測(cè)試一:在子組件的props未發(fā)生任何變更的情況下是否會(huì)發(fā)生重新渲染

import React, { useState } from "react"
import { Select } from "antd"
const { Option } = Select
export function Chart() {
? ? console.log("父組件重新渲染了")
? ? const [selectedOption, setSelectedOption] = useState("option1")
? ? function handleSelectChange(value) {
? ? ? ? setSelectedOption(value)
? ? }
? ? return (
? ? ? ? <div>
? ? ? ? ? ? <Select defaultValue={selectedOption} onChange={handleSelectChange}>
? ? ? ? ? ? ? ? <Option value="option1">Option 1</Option>
? ? ? ? ? ? ? ? <Option value="option2">Option 2</Option>
? ? ? ? ? ? ? ? <Option value="option3">Option 3</Option>
? ? ? ? ? ? </Select>
? ? ? ? ? ? <ChildComponent />
? ? ? ? </div>
? ? )
}
function ChildComponent() {
? ? console.log("ChildComponent重新渲染了")
? ? return <div>Hello from ChildComponent</div>
}

測(cè)試結(jié)論:子組件即使不接收任何props當(dāng)父組件重新渲染時(shí)子組件也會(huì)重新渲染

解決方案

使用React.memo,關(guān)于memo介紹如下:

包裝一個(gè)組件memo以獲得該組件的記憶版本。只要組件的 props 沒有改變,當(dāng)它的父組件重新渲染時(shí),組件的這個(gè)記憶版本通常不會(huì)被重新渲染。

但 React 可能仍然會(huì)重新渲染它:memoization 是一種性能優(yōu)化,而不是保證。

import React, { useState } from "react"
import { Select } from "antd"
const { Option } = Select
export function Chart() {
? ? console.log("父組件重新渲染了")
? ? const [selectedOption, setSelectedOption] = useState("option1")
? ? function handleSelectChange(value) {
? ? ? ? setSelectedOption(value)
? ? }
? ? return (
? ? ? ? <div>
? ? ? ? ? ? <Select defaultValue={selectedOption} onChange={handleSelectChange}>
? ? ? ? ? ? ? ? <Option value="option1">Option 1</Option>
? ? ? ? ? ? ? ? <Option value="option2">Option 2</Option>
? ? ? ? ? ? ? ? <Option value="option3">Option 3</Option>
? ? ? ? ? ? </Select>
? ? ? ? ? ? {/* <ChildComponent /> */}
? ? ? ? ? ? <MemoChild />
? ? ? ? </div>
? ? )
}
function ChildComponent() {
? ? console.log("ChildComponent重新渲染了")
? ? return <div>Hello from ChildComponent</div>
}
const MemoChild = React.memo(ChildComponent)

即使使用memo,但其仍然會(huì)在其接受的props發(fā)生變更時(shí)及其所在的上下文發(fā)生變更時(shí)重新渲染。

因此為了更好的使用memo,應(yīng)當(dāng)盡量減少props的變化。具體方案如下:

  • 當(dāng)props為引用值時(shí)應(yīng)當(dāng)使用useMemo返回相同引用值
  • 使得 props 中接受最少的必要信息
  • 當(dāng)props包含函數(shù)時(shí),要么在組件外部聲明它以使其永遠(yuǎn)不會(huì)更改,要么useCallback在重新渲染之間緩存其定義。

例如:當(dāng)props為對(duì)象時(shí),應(yīng)當(dāng)減少props對(duì)象的變化

測(cè)試內(nèi)組件對(duì)memo的影響

接下來進(jìn)行另外的測(cè)試,將子組件放在Adapt中稱為內(nèi)部組件

可以發(fā)現(xiàn),此時(shí)子組件的緩存失效了,盡管子組件沒有props的變化

但每次父組件重新渲染子組件仍然會(huì)重新渲染

import React, { useState } from "react"
import { Select } from "antd"
const { Option } = Select
export function Adapt() {
? ? console.log("父組件重新渲染了")
? ? const [selectedOption, setSelectedOption] = useState("option1")
? ? function handleSelectChange(value) {
? ? ? ? setSelectedOption(value)
? ? }
? ? return (
? ? ? ? <div>
? ? ? ? ? ? <Select defaultValue={selectedOption} onChange={handleSelectChange}>
? ? ? ? ? ? ? ? <Option value="option1">Option 1</Option>
? ? ? ? ? ? ? ? <Option value="option2">Option 2</Option>
? ? ? ? ? ? ? ? <Option value="option3">Option 3</Option>
? ? ? ? ? ? </Select>
? ? ? ? ? ? {/* <ChildComponent /> */}
? ? ? ? ? ? <MemoChild />
? ? ? ? </div>
? ? )
}
function ChildComponent() {
? ? console.log("ChildComponent重新渲染了")
? ? return <div>Hello from ChildComponent</div>
}
const MemoChild = React.memo(ChildComponent)

React.memo 失效的原因是因?yàn)閷?ChildComponent 和 MemoChild 定義在了 Adapt 組件的內(nèi)部。這意味著每當(dāng) Adapt 組件重新渲染時(shí),ChildComponent 和 MemoChild 都會(huì)重新定義,導(dǎo)致了緩存失效。

測(cè)試props變更對(duì)memo的影響

改寫測(cè)試用例,測(cè)試當(dāng)父組件的props變更時(shí),子組件是否重新渲染

import React, { useState } from "react"
import { Select } from "antd"
const { Option } = Select
export function Adapt() {
? ? console.log("Adapt重新渲染了")
? ? const [filterList, setFilterList] = useState(null)
? ? function onRefresh(value) {
? ? ? ? setFilterList(value)
? ? }
? ? return (
? ? ? ? <div>
? ? ? ? ? ? <Chart {...{ onRefresh, filterList }} />
? ? ? ? </div>
? ? )
}
function Chart({ filterList, onRefresh }) {
? ? console.log("Chart重新渲染了")
? ? return (
? ? ? ? <div>
? ? ? ? ? ? <Select value={filterList} onChange={onRefresh}>
? ? ? ? ? ? ? ? <Option value="option1">Option 1</Option>
? ? ? ? ? ? ? ? <Option value="option2">Option 2</Option>
? ? ? ? ? ? ? ? <Option value="option3">Option 3</Option>
? ? ? ? ? ? </Select>
? ? ? ? ? ? <MemoChild />
? ? ? ? </div>
? ? )
}
function ChildComponent() {
? ? console.log("ChildComponent重新渲染了")
? ? return <div>hello</div>
}
const MemoChild = React.memo(ChildComponent)
export default Adapt

子組件仍然可以進(jìn)行正常的緩存,可得知當(dāng)子組件使用memo

無論是父組件的state或者props狀態(tài)變更,只要子組件的props和上下文未變化都不會(huì)影響到子組件

問題結(jié)論

導(dǎo)致組件重復(fù)渲染的可能如下:

  • 當(dāng)子組件未使用memo時(shí),子組件會(huì)隨著父組件一起重新渲染,因此即使子組件的props未變化仍會(huì)多次渲染子組件
  • 當(dāng)子組件使用了memo,但仍然重新渲染
    • 子組件接受的props或者context變更了
    • 子組件的上下文及props都沒變更,但子組件時(shí)內(nèi)部組件所以每次都是重新創(chuàng)建的

因此為了優(yōu)化子組件的多次渲染應(yīng)當(dāng)注意:

  • 重復(fù)的渲染可以使用memo緩存
  • 父組件傳遞的重復(fù)的props參數(shù)可以使用useMemo
  • 每次傳遞的props應(yīng)當(dāng)是最小的props,不然無關(guān)的props變更會(huì)影響組件緩存失效
  • 組件及函數(shù)應(yīng)該盡量以純函數(shù)形式寫在組件之外,避免組件重渲染時(shí)重新創(chuàng)建函數(shù)帶來的影響

 到此這篇關(guān)于react組件多次渲染問題的解決的文章就介紹到這了,更多相關(guān)react組件多次渲染內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • React淺析Fragments使用方法

    React淺析Fragments使用方法

    這篇文章主要介紹了React Fragments使用方法,關(guān)于react Fragments,React中一個(gè)常見模式是為一個(gè)組件返回多個(gè)元素。Fragments 可以讓你聚合一個(gè)子元素列表,并且不在DOM中增加額外節(jié)點(diǎn)
    2022-12-12
  • 手把手帶你用React擼一個(gè)日程組件

    手把手帶你用React擼一個(gè)日程組件

    這篇文章主要給大家介紹了關(guān)于利用React擼一個(gè)日程組件的相關(guān)資料,包括日常組件的實(shí)現(xiàn)思路、使用的技術(shù)、以及遇到的技術(shù)難點(diǎn),并給提供了詳細(xì)的實(shí)例代碼,需要的朋友可以參考下
    2021-07-07
  • React 組件中實(shí)現(xiàn)事件代理

    React 組件中實(shí)現(xiàn)事件代理

    React的事件系統(tǒng)和瀏覽器事件系統(tǒng)相比,主要增加了兩個(gè)特性:事件代理、和事件自動(dòng)綁定,本文主要介紹了React 組件中實(shí)現(xiàn)事件代理,具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-01-01
  • react中useEffect Hook作用小結(jié)

    react中useEffect Hook作用小結(jié)

    React中useEffect是一個(gè)非常重要的Hook,它允許我們函數(shù)組件中執(zhí)行副作用操作,本文就來介紹一下useEffect Hook作用,具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-11-11
  • Ant Design組件庫(kù)的使用教程

    Ant Design組件庫(kù)的使用教程

    AntDesign ,簡(jiǎn)稱antd是基于 Ant Design 設(shè)計(jì)體系的 React UI 組件庫(kù),主要用于研發(fā)企業(yè)級(jí)中后臺(tái)產(chǎn)品,這篇文章主要介紹了Ant Design組件庫(kù)的使用教程,需要的朋友可以參考下
    2023-12-12
  • React Hook 'useEffect' is called in function報(bào)錯(cuò)解決

    React Hook 'useEffect' is call

    這篇文章主要為大家介紹了React Hook 'useEffect' is called in function報(bào)錯(cuò)解決,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-12-12
  • useCallback和useMemo的正確用法詳解

    useCallback和useMemo的正確用法詳解

    這篇文章主要為大家介紹了useCallback和useMemo的正確用法詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-01-01
  • React中useEffect函數(shù)的使用詳解

    React中useEffect函數(shù)的使用詳解

    useEffect是React中的一個(gè)鉤子函數(shù),用于處理副作用操作,這篇文章主要為大家介紹了React中useEffect函數(shù)的具體用法,希望對(duì)大家有所幫助
    2023-08-08
  • React組件內(nèi)事件傳參實(shí)現(xiàn)tab切換的示例代碼

    React組件內(nèi)事件傳參實(shí)現(xiàn)tab切換的示例代碼

    本篇文章主要介紹了React組件內(nèi)事件傳參實(shí)現(xiàn)tab切換的示例代碼,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2018-07-07
  • React深入了解原理

    React深入了解原理

    React是用于構(gòu)建用戶界面的JavaScript庫(kù),?[1]??起源于Facebook的內(nèi)部項(xiàng)目,該公司對(duì)市場(chǎng)上所有?JavaScript?MVC框架都不滿意,決定自行開發(fā)一套,用于架設(shè)Instagram的網(wǎng)站
    2022-07-07

最新評(píng)論