react數據管理中的setState與Props詳解
setState調用原理
setState
是 React 中用于更新組件狀態(tài)(state)的方法。它的調用原理可以分為以下幾個步驟:
- 狀態(tài)的改變:當調用
setState
時,React 會將新的狀態(tài)對象與當前狀態(tài)對象進行合并(合并過程是淺合并)。React 不會直接修改當前狀態(tài)對象,而是創(chuàng)建一個新的狀態(tài)對象,以確保不直接修改狀態(tài),從而遵循 React 的不可變性原則。 - 觸發(fā)重新渲染:一旦狀態(tài)更新完成,React 會調用組件的
render
方法來重新渲染組件。在重新渲染過程中,React 會生成一個新的虛擬 DOM 樹(Virtual DOM)。 - 虛擬 DOM 比較:React 會將新生成的虛擬 DOM 樹與之前的虛擬 DOM 樹進行比較,找出兩者之間的差異(所謂的變更集合)。
- 局部更新:React 將變更集合中的差異應用到實際的 DOM 中,以更新用戶界面。這是一個高效的過程,因為 React 僅更新了需要更新的部分,而不是重新渲染整個頁面。
- 生命周期方法調用:在更新完成后,React 會調用適當的生命周期方法,如
componentDidUpdate
,以便開發(fā)者可以執(zhí)行一些操作,例如獲取最新的 DOM 元素引用或執(zhí)行副作用操作。
需要注意的是,由于 setState
是異步的,React 可能會將多次的 setState
調用合并成一次更新,以提高性能。這意味著在一個函數內多次調用 setState
可能不會導致多次重新渲染,而是在函數執(zhí)行結束后一次性更新狀態(tài)和重新渲染。
例如,以下代碼中的多次 setState
調用會被合并成一次更新:
this.setState({ count: this.state.count + 1 }); this.setState({ count: this.state.count + 1 });
這個特性有時會導致不直觀的行為,因此可以使用函數式的 setState
形式來確保狀態(tài)更新是基于先前的狀態(tài)的,而不受合并的影響:
this.setState((prevState) => ({ count: prevState.count + 1 }));
setState第二個參數
this.setState(newState, callback);
newState
是一個對象或函數,用于描述要更新的狀態(tài)。這可以是一個新的狀態(tài)值或一個函數,該函數接受前一個狀態(tài)作為參數,返回新的狀態(tài)。callback
是一個可選的回調函數,它會在狀態(tài)更新完成后被調用。
例如:
import React, { Component } from 'react'; class Counter extends Component { constructor(props) { super(props); this.state = { count: 0, }; } handleIncrement = () => { this.setState({ count: this.state.count + 1 }, () => { // 回調函數,會在狀態(tài)更新后被調用 console.log('Count updated:', this.state.count); }); } render() { return ( <div> <p>Count: {this.state.count}</p> <button onClick={this.handleIncrement}>Increment</button> </div> ); } }
在上述示例中,當點擊按鈕并調用 handleIncrement
方法時,this.setState
用于增加 count
的值。回調函數通過 console.log
輸出更新后的 count
值,這是在狀態(tài)更新后被調用的。
回調函數的主要用途包括:
- 執(zhí)行需要在狀態(tài)更新后立即執(zhí)行的代碼。
- 執(zhí)行副作用操作,如發(fā)起網絡請求或操作 DOM。
- 在狀態(tài)更新后通知其他組件或觸發(fā)其他操作。
props和state區(qū)別
props
(屬性)和 state
(狀態(tài))是 React 中用于管理組件數據的兩個不同概念,它們有以下主要區(qū)別:
來源:
props
:是由父組件傳遞給子組件的數據。父組件通過屬性(props)將數據傳遞給子組件,子組件不可以直接修改這些數據,它們是只讀的。state
:是組件內部維護的數據,用于描述組件的特定狀態(tài)。組件可以自己管理和修改自己的狀態(tài)。
可變性:
props
:是不可變的(immutable),子組件不能直接修改從父組件接收的 props。Props 用于從外部傳遞信息給組件,組件應該將 props 視為只讀數據。state
:是可變的(mutable),組件可以通過調用setState
方法來更新自己的狀態(tài)。
管理:
props
:由父組件管理和傳遞,子組件只能訪問和使用 props。state
:由組件自己管理和維護,組件可以在需要時修改自己的狀態(tài)。
作用:
props
:用于傳遞數據給子組件,以控制子組件的行為和顯示。state
:用于管理組件內部的狀態(tài),以響應用戶交互、數據請求、或其他事件。
傳遞:
props
:通過組件的屬性(props)傳遞給子組件。父組件可以通過屬性來控制子組件的行為。state
:在組件內部聲明和管理,可以通過setState
方法來更新。
默認值:
props
:可以為 props 設置默認值,以防止未傳遞某個屬性時出現錯誤。state
:可以在組件的構造函數中設置初始狀態(tài)(state)的默認值。
更新:
props
:當父組件的 props 發(fā)生變化時,會觸發(fā)子組件的重新渲染,子組件會接收新的 props。state
:當組件的狀態(tài)(state)發(fā)生變化時,會觸發(fā)組件的重新渲染,從而更新界面。
props改變后如何更新組件
class組件
- 父組件傳遞新的
props
:父組件可以通過修改傳遞給子組件的props
數據來引發(fā)子組件的更新。這可以通過在父組件中修改props
值或通過父組件的狀態(tài)變化來實現。 - 子組件的
componentWillReceiveProps
(已廢棄,不推薦使用)、getDerivedStateFromProps
或componentDidUpdate
方法:當子組件接收到新的props
后,React 將觸發(fā)這些生命周期方法之一,具體取決于 React 版本和組件實現。
在 React 16.3 及更早版本中,可以使用 componentWillReceiveProps
生命周期方法來處理新的 props
。
在 React 16.3 及以后的版本中,推薦使用 getDerivedStateFromProps
靜態(tài)方法或 componentDidUpdate
來處理新的 props
。
在生命周期方法中更新組件狀態(tài)或執(zhí)行其他操作:在上述生命周期方法中,可以訪問新的 props
和組件當前的狀態(tài)(this.props
和 this.state
)以及之前的 props
和 state
。可以根據新的 props
數據來更新組件的狀態(tài),從而觸發(fā)重新渲染。
例如,使用 getDerivedStateFromProps來處理新的 props:
class MyComponent extends React.Component { static getDerivedStateFromProps(nextProps, prevState) { // 檢查新的 props,并根據需要更新狀態(tài) if (nextProps.someProp !== prevState.someProp) { return { someState: nextProps.someProp, }; } return null; // 不更新狀態(tài) } render() { // 渲染組件 return <div>{this.state.someState}</div>; } }
或者componentDidUpdate:
import React, { Component } from 'react'; class MyComponent extends Component { state = { count: 0, }; componentDidUpdate(prevProps, prevState) { if (this.props.someProp !== prevProps.someProp) { console.log('Props changed:', prevProps.someProp, '->', this.props.someProp); } } render() { return <div>{this.props.someProp}</div>; } } export default MyComponent;
React 16.3 及以后的版本,componentWillReceiveProps
生命周期方法已經被標記為已廢棄,不再推薦使用。相反,推薦使用 getDerivedStateFromProps
或 componentDidUpdate
來處理 props
的變化。
函數組件
在函數組件中,可以使用 React 的鉤子函數來判斷組件何時更新,特別是 useEffect
鉤子。下面是如何在函數組件中判斷組件何時更新:
1.使用 useEffect
鉤子:使用useEffect
鉤子可以函數組件中執(zhí)行副作用操作,并且可以根據依賴項來判斷何時觸發(fā)這些副作用??梢詫?props
或其他狀態(tài)值作為依賴項,當這些依賴項發(fā)生變化時,useEffect
中的代碼將被執(zhí)行。
import React, { useEffect } from 'react'; function MyComponent(props) { // 使用 useEffect 鉤子來判斷何時更新 useEffect(() => { // 這里的代碼在組件每次渲染后都會執(zhí)行 // 可以在這里根據新的 props 進行操作 console.log('Props updated:', props.someProp); }, [props.someProp]); // 僅在 props.someProp 發(fā)生變化時執(zhí)行 return <div>{props.someProp}</div>; }
在上面的示例中,我們將 props.someProp
作為 useEffect
的依賴項,因此當 props.someProp
發(fā)生變化時,useEffect
中的代碼將被執(zhí)行,從而可以判斷組件何時更新。
2.使用 React.memo
(可選):如果希望函數組件在特定 props
變化時才進行更新,并且不關心其他的 props
,可以使用 React.memo
來包裝函數組件。這將創(chuàng)建一個經過優(yōu)化的組件,只有在指定的 props
發(fā)生變化時才會觸發(fā)重新渲染。
import React from 'react'; function MyComponent(props) { return <div>{props.someProp}</div>; } // 使用 React.memo 包裝組件,只有 someProp 變化時才重新渲染 export default React.memo(MyComponent);
到此這篇關于react數據管理之setState與Props的文章就介紹到這了,更多相關react setState與Props內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
useEffect如何通過form.getFieldValue(‘xxx‘)監(jiān)聽Form表單變化
這篇文章主要介紹了useEffect如何通過form.getFieldValue(‘xxx‘)監(jiān)聽Form表單變化問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-03-03