摘要:組件狀態(tài)是一種持有,處理和使用信息的方式。更新唯一你能直接寫的地方應(yīng)該是組件的構(gòu)造函數(shù)中。是異步的事實(shí)上會(huì)引起的一致性處理重新渲染組件樹的過程,是下一個(gè)屬性的基礎(chǔ)即是異步的。常見錯(cuò)誤其中最常見的錯(cuò)誤之一就是在構(gòu)造函數(shù)中使用設(shè)置的值。
組件狀態(tài)(state)是一種持有,處理和使用信息的方式。state包含的信息僅作用于一個(gè)給定組件的內(nèi)部,并允許你根據(jù)它實(shí)現(xiàn)組件的一些邏輯。state通常是一個(gè)POJO(Plain Old Java[Script] Object)對(duì)象,改變它是使得組件重新render自己的方式之一。
state是react背后原理的重要基礎(chǔ)概念之一,但是它也有一些特點(diǎn)使得它用起來會(huì)有點(diǎn)難以捉摸并且有可能會(huì)導(dǎo)致在你的應(yīng)用中出現(xiàn)一些預(yù)料之外的行為。
更新State
唯一你能直接寫this.state的地方應(yīng)該是組件的構(gòu)造函數(shù)中。在其它所有地方你都應(yīng)該使用this.setState函數(shù),它接受一個(gè)對(duì)象作為參數(shù),這個(gè)對(duì)象最終會(huì)被合并到組件的當(dāng)前狀態(tài)中。
而在技術(shù)上你是可以通過this.state={//a new object}這種方式直接修改狀態(tài)的,但是它不會(huì)引起組件使用新的值去重新渲染,然后導(dǎo)致狀態(tài)不一致的問題。
setState是異步的
事實(shí)上setState會(huì)引起的一致性處理(reconciliation)——重新渲染組件樹的過程,是下一個(gè)屬性的基礎(chǔ)即setState是異步的。這允許我們?cè)趩蝹€(gè)作用域內(nèi)多次調(diào)用setState而不會(huì)觸發(fā)不必要的重新渲染整個(gè)樹。
這就是為什么在你更新state后并不能立馬看見新的值。
// 假設(shè) this.state = { value: 0 } this.setState({ value: 1 }); console.log(this.state.value); // 0
React 也會(huì)嘗試將多次setState調(diào)用組合或者批處理為一次調(diào)用:
// 假設(shè) this.state = { value: 0 }; this.setState({ value: this.state.value + 1}); this.setState({ value: this.state.value + 1}); this.setState({ value: this.state.value + 1});
在上面所有的調(diào)用完成后,this.state.value將是1,而不是我們的期望值3。那么怎么得到期望值3呢?
setState接受一個(gè)函數(shù)作為它的參數(shù)
如果你傳遞一個(gè)函數(shù)作為setState的第一個(gè)參數(shù),React將會(huì)使用在當(dāng)前調(diào)用時(shí)刻的state去調(diào)用它并期望你返回一個(gè)對(duì)象合并到state中。所以你可以把我們上面的代碼改成下面這樣即可:
// 假設(shè) this.state = { value: 0 }; this.setState((state) => ({ value: state.value + 1})); this.setState((state) => ({ value: state.value + 1})); this.setState((state) => ({ value: state.value + 1}));
這樣this.state.value 的值就是 3了,就和上面我們認(rèn)為期望值應(yīng)該是3相一致了。
記住當(dāng)在更新state為一個(gè)值的時(shí)候應(yīng)始終使用這種語法,它的計(jì)算是基于前面的一個(gè)狀態(tài)的。
setState是同步的嗎???
記住你剛才學(xué)習(xí)到setState是異步的。事實(shí)證明并非始終如此。這取決于執(zhí)行上下文,請(qǐng)看下面的例子:
render() { return } inc() { console.log("before: " + this.state.test); this.setState({ test: this.state.test+1 }); console.log("after: " + this.state.test); }
點(diǎn)擊按鈕元素將會(huì)導(dǎo)致你的console中顯示:
// click! before: 1 after: 1 // click! before: 2 after: 2
但如果我們添加以下代碼:
componentDidMount() { setInterval(this.inc, 1000); }
我們將在console中看到:
before: 1 after: 2 before: 2 after: 3
因此,我們需要學(xué)習(xí)什么時(shí)候期望得到哪種行為嗎?并不是如此。假設(shè)setState的確是異步是非常安全的,因?yàn)樵谖磥硭褪侨绱恕?/p>
setState接受一個(gè)回調(diào)函數(shù)
如果你需要執(zhí)行一些函數(shù),或者驗(yàn)證狀態(tài)是否真的有更新正確。你還可以給setState傳遞一個(gè)函數(shù)作為第二個(gè)參數(shù),這個(gè)函數(shù)會(huì)在狀態(tài)更新完畢后得到執(zhí)行。請(qǐng)記住由于單個(gè)塊內(nèi)的所有更新會(huì)被合并成一個(gè),這將導(dǎo)致每個(gè)setState中的回調(diào)中得到的state值是全更新的state。
另外一種可以保證你的代碼執(zhí)行是在更新完成以后的方式是將執(zhí)行代碼放在componentWillUpdate?或者?componentDidUpdate中。然而對(duì)比回調(diào)函數(shù)的方式,這兩個(gè)方法會(huì)在shouldComponentUpdate中阻止你的組件更新時(shí)不會(huì)被調(diào)用。
常見錯(cuò)誤
其中最常見的錯(cuò)誤之一就是在構(gòu)造函數(shù)中使用props設(shè)置state的值。考慮如下代碼:
class Component extends React.Component { constructor(props) { super(props); this.state = { value: this.props.value }; } render() { returnThe value is: {this.state.value}} }
如果它的父組件這樣render它:
它將會(huì)正確渲染value為42,但如果父組件中修改成如下:
那它仍會(huì)認(rèn)為this.state.value是42,這是因?yàn)镽eact并不會(huì)銷毀組件并重新創(chuàng)建它——它會(huì)重用一旦渲染好的組件,并且不會(huì)重新執(zhí)行構(gòu)造函數(shù)。要避免這個(gè)問題,你應(yīng)該不要將props賦值給state,而應(yīng)在render方法中使用this.props.value。
如果你還是想要使用state(如果你的props是以一種非常復(fù)雜的計(jì)算的使用模式,你不希望每一次render都執(zhí)行這些復(fù)雜的計(jì)算),你還可以實(shí)現(xiàn)一種在需要的時(shí)候才去更新state的解決方案,例如:
class Component extends React.Component { constructor(props) { super(props); this.state = { value: this.props.value }; } componentWillReceiveProps(nextProps) { if(nextProps.value !== this.props.value) { this.setState({value: nextProps.value}); } } render() { returnThe value is: {this.state.value}} }
請(qǐng)記住任何componentWill*函數(shù)都不是一個(gè)合適的地方去觸發(fā)side effect(如ajax請(qǐng)求),所以請(qǐng)使用
componentDidUpdate(previousProps, previousState),同樣也提供和上面類似的if防護(hù)語句確保在沒有變化時(shí)不會(huì)執(zhí)行相關(guān)代碼。
寫在最后的話:這篇文章在Medium中獲得超過2.1K的贊,挺不錯(cuò)的,值得翻譯!
翻譯如有不正,歡迎留言指出!
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://m.specialneedsforspecialkids.com/yun/96064.html
摘要:原文地址由于只涉及層的處理,所以構(gòu)建大型應(yīng)用應(yīng)該搭配一個(gè)框架模式才能使后期維護(hù)成本相對(duì)較小正是官方給出的應(yīng)用架構(gòu),他推崇一種單向的數(shù)據(jù)流動(dòng)模式,看下圖感受下整個(gè)流程是用戶與層交互,觸發(fā)使用進(jìn)行分發(fā)觸發(fā)回調(diào)進(jìn)行更新更新觸發(fā)層事件層收到信號(hào)進(jìn) 原文地址:https://gmiam.com/post/react-... 由于 React 只涉及 UI 層的處理,所以構(gòu)建大型應(yīng)用應(yīng)該搭配一個(gè)框...
摘要:一般這種情況會(huì)在類的構(gòu)造函數(shù)內(nèi)創(chuàng)建一個(gè)屬性,引用或詞法域的,但后面會(huì)看到我們有更好的辦法,避免這種手工代碼。 換句話說,StateUp模式把面向?qū)ο蟮脑O(shè)計(jì)方法應(yīng)用到了狀態(tài)對(duì)象的管理上,在遵循React的組件化機(jī)制和基于props實(shí)現(xiàn)組件通訊方式的前提之下做到了這一點(diǎn)。 ---- 少婦白潔 閱讀本文之前,請(qǐng)確定你讀過React的官方文檔中關(guān)于Lifting State Up的論述: ht...
摘要:第一次了解這項(xiàng)特性的時(shí)候,真的有一種豁然開朗,發(fā)現(xiàn)新大陸的感覺。為了解決這一痛點(diǎn),才會(huì)有剪頭函數(shù)的綁定特性。它同時(shí)具備和三個(gè)生命周期函數(shù)的執(zhí)行時(shí)機(jī)。 歡迎關(guān)注我的公眾號(hào)睿Talk,獲取我最新的文章:showImg(https://segmentfault.com/img/bVbmYjo); 一、前言 React Hooks 是從 v16.8 引入的又一開創(chuàng)性的新特性。第一次了解這項(xiàng)特性...
摘要:本文翻譯自原作者如果有任何版權(quán)問題,請(qǐng)聯(lián)系當(dāng)你在組件中調(diào)用時(shí),你覺得會(huì)發(fā)生什么當(dāng)然,會(huì)用這條狀態(tài)重新渲染組件并且更新匹配到的,然后返回元素。如果你之前使用過一些渲染器比如說,你可能知道在頁面中使用超過一個(gè)渲染器是沒什么問題的。 本文翻譯自:How Does setState Know What to Do?原作者:Dan Abramov 如果有任何版權(quán)問題,請(qǐng)聯(lián)系shuirong199...
摘要:右側(cè)展現(xiàn)對(duì)應(yīng)產(chǎn)品。我們使用命名為的對(duì)象表示過濾條件信息,如下此數(shù)據(jù)需要在組件中進(jìn)行維護(hù)。因?yàn)榻M件的子組件和都將依賴這項(xiàng)數(shù)據(jù)狀態(tài)。化應(yīng)用再回到之前的場(chǎng)景,我們?cè)O(shè)計(jì)化函數(shù),進(jìn)一步可以簡(jiǎn)化為對(duì)于的偏應(yīng)用即上面提到的相信大家已經(jīng)理解了這么做的好處。 showImg(https://segmentfault.com/img/remote/1460000014458612?w=1240&h=663...
閱讀 2117·2023-04-26 00:50
閱讀 2488·2021-10-13 09:39
閱讀 2221·2021-09-22 15:34
閱讀 1613·2021-09-04 16:41
閱讀 1343·2019-08-30 15:55
閱讀 2441·2019-08-30 15:53
閱讀 1714·2019-08-30 15:52
閱讀 754·2019-08-29 16:19