組件的state和setState

state


Component的顯示形態(tài)是有它的數(shù)據(jù)狀態(tài)和參數(shù)決定的一罩。React.js的state就是用來(lái)存儲(chǔ)這種可變化的狀態(tài)。
例如以下代碼:

import React, { Component } from 'react'
import ReactDOM from 'react-dom'
import './index.css'

class LikeButton extends Component {
  constructor () {
    super()
    this.state = { isLiked: false }
  }

  handleClickOnLikeButton () {
    this.setState({
      isLiked: !this.state.isLiked
    })
  }

  render () {
    return (
      <button onClick={this.handleClickOnLikeButton.bind(this)}>
        {this.state.isLiked ? '取消' : '點(diǎn)贊'} 
      </button>
    )
  }
}
···

isLiked被存放在實(shí)例的state對(duì)象中植榕,這個(gè)對(duì)象在構(gòu)造函數(shù)里被初始化。這個(gè)Component的render方法里坯认,會(huì)根據(jù)state中isLiked的不同顯示‘取消’或者‘點(diǎn)贊’盾似,并且button加上了點(diǎn)擊的事件監(jiān)聽(tīng)。

在 handleClickOnLikeButton 事件監(jiān)聽(tīng)方法中皆看,使用setState方法仓坞,每次點(diǎn)擊會(huì)更新isLiked的屬性為!isLiked,這樣就完成了點(diǎn)贊和取消腰吟。

setState方法是由父類(lèi)Component提供的无埃。當(dāng)我們調(diào)用這個(gè)函數(shù)的時(shí)候,React.js會(huì)更新組件的狀態(tài)state毛雇,并且重新調(diào)用render方法嫉称,更新顯示頁(yè)面內(nèi)容。

注意灵疮,不能直接使用 this.state = xxx 的方式來(lái)修改织阅,如果這樣做React.js就沒(méi)法知道你修改了的組件的狀態(tài),也就沒(méi)法刷新頁(yè)面震捣。所以荔棉,一定要使用setState的方法,它接受一個(gè)對(duì)象或者函數(shù)作為參數(shù)蒿赢。

傳入一個(gè)對(duì)象的時(shí)候润樱,只需要傳遞要更新的屬性就可以了。例如:

  constructor (props) {
    super(props)
    this.state = {
      name: 'Sam',
      isLiked: false
    }
  }

  handleClickOnLikeButton () {
    this.setState({
      isLiked: !this.state.isLiked
    })
  }

handleClickOnLikedButton方法里不需要改嗎state中的name屬性羡棵,因此在setState方法中是需要傳入isLiked屬性就可以了壹若。

setState接受函數(shù)作為參數(shù)


當(dāng)你調(diào)用setState方法時(shí),React.js并不會(huì)馬上修改state皂冰,而是把這個(gè)對(duì)象放到一個(gè)更新隊(duì)列中店展,稍后才會(huì)從隊(duì)列中把新的狀態(tài)提取出來(lái)合并到state當(dāng)中,然后再出發(fā)組件更新秃流。這一點(diǎn)要特別注意赂蕴,可以體會(huì)一下以下代碼:

  handleClickOnLikeButton () {
    console.log(this.state.isLiked)//false
    this.setState({
      isLiked: !this.state.isLiked
    })
    console.log(this.state.isLiked)//false
  }

此時(shí)控制臺(tái)會(huì)輸出兩次false,即使中間調(diào)用了一次setState方法舶胀,也沒(méi)有什么bug概说,但是isLiked得值并沒(méi)有改變。只是React.js得setState把傳進(jìn)來(lái)的狀態(tài)緩存了峻贮,沒(méi)有立即更新席怪,所以第二次打印時(shí),還是原來(lái)的isLiked纤控。

所以如果想在setState之后使用新的state來(lái)做后續(xù)的運(yùn)算就做不到了挂捻,例如:

  handleClickOnLikeButton () {
    this.setState({ count: 0 }) // => this.state.count 還是 undefined
    this.setState({ count: this.state.count + 1}) // => undefined + 1 = NaN
    this.setState({ count: this.state.count + 2}) // => NaN + 2 = NaN
  }

上面的代碼的運(yùn)行結(jié)果并不能達(dá)到我們的預(yù)期,我們希望 count 運(yùn)行結(jié)果是 3 船万,可是最后得到的是 NaN刻撒。但是這種后續(xù)操作依賴(lài)前一個(gè) setState 的結(jié)果的情況并不罕見(jiàn)骨田。

來(lái)看第二種setState方法的第二種使用方式,可以傳入一個(gè)函數(shù)作為參數(shù)声怔。React.js會(huì)把上一個(gè)setState的結(jié)果傳入這個(gè)函數(shù)态贤,你就可以使用該結(jié)果進(jìn)行運(yùn)算、操作醋火,然后返回一個(gè)對(duì)象為更新state對(duì)象:

  handleClickOnLikeButton () {
    this.setState((prevState) => {
      return { count: 0 }
    })
    this.setState((prevState) => {
      return { count: prevState.count + 1 } // 上一個(gè) setState 的返回是 count 為 0悠汽,當(dāng)前返回 1
    })
    this.setState((prevState) => {
      return { count: prevState.count + 2 } // 上一個(gè) setState 的返回是 count 為 1,當(dāng)前返回 3
    })
    // 最后的結(jié)果是 this.state.count 為 3
  }

這樣就可以達(dá)到上述的利用上一次 setState 結(jié)果進(jìn)行運(yùn)算的效果芥驳。

setState合并


上面進(jìn)行了三次setState柿冲,但實(shí)際上組件只會(huì)重新渲染一次,而不是三次兆旬。這是因?yàn)镽eact.js內(nèi)部會(huì)把JavaScript事件循環(huán)中的消息隊(duì)列的同一個(gè)消息中的setState都進(jìn)行合并以后再重新渲染組件假抄。

深層的原理我也沒(méi)搞太清楚,現(xiàn)階段只要記桌鲡:在使用React.js的時(shí)候宿饱,并不需要擔(dān)心多次進(jìn)行setState會(huì)帶來(lái)性能問(wèn)題。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末脚祟,一起剝皮案震驚了整個(gè)濱河市谬以,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌愚铡,老刑警劉巖蛉签,帶你破解...
    沈念sama閱讀 217,907評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件胡陪,死亡現(xiàn)場(chǎng)離奇詭異沥寥,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)柠座,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,987評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)邑雅,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人妈经,你說(shuō)我怎么就攤上這事淮野。” “怎么了吹泡?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,298評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵骤星,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我爆哑,道長(zhǎng)洞难,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,586評(píng)論 1 293
  • 正文 為了忘掉前任揭朝,我火速辦了婚禮队贱,結(jié)果婚禮上色冀,老公的妹妹穿的比我還像新娘。我一直安慰自己柱嫌,他們只是感情好锋恬,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,633評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著编丘,像睡著了一般与学。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上嘉抓,一...
    開(kāi)封第一講書(shū)人閱讀 51,488評(píng)論 1 302
  • 那天癣防,我揣著相機(jī)與錄音,去河邊找鬼掌眠。 笑死蕾盯,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的蓝丙。 我是一名探鬼主播级遭,決...
    沈念sama閱讀 40,275評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼渺尘!你這毒婦竟也來(lái)了挫鸽?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 39,176評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤鸥跟,失蹤者是張志新(化名)和其女友劉穎丢郊,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體医咨,經(jīng)...
    沈念sama閱讀 45,619評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡枫匾,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,819評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了拟淮。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片干茉。...
    茶點(diǎn)故事閱讀 39,932評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖很泊,靈堂內(nèi)的尸體忽然破棺而出角虫,到底是詐尸還是另有隱情,我是刑警寧澤委造,帶...
    沈念sama閱讀 35,655評(píng)論 5 346
  • 正文 年R本政府宣布戳鹅,位于F島的核電站,受9級(jí)特大地震影響昏兆,放射性物質(zhì)發(fā)生泄漏枫虏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,265評(píng)論 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望模软。 院中可真熱鬧伟骨,春花似錦、人聲如沸燃异。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,871評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)回俐。三九已至逛腿,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間仅颇,已是汗流浹背单默。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,994評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留忘瓦,地道東北人搁廓。 一個(gè)月前我還...
    沈念sama閱讀 48,095評(píng)論 3 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像耕皮,于是被迫代替她去往敵國(guó)和親境蜕。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,884評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • React.js 小書(shū) Lesson10 - 組件的 state 和 setState 本文作者:胡子大哈本文原文...
    胡子大哈閱讀 424評(píng)論 0 1
  • 作為一個(gè)合格的開(kāi)發(fā)者凌停,不要只滿(mǎn)足于編寫(xiě)了可以運(yùn)行的代碼粱年。而要了解代碼背后的工作原理;不要只滿(mǎn)足于自己的程序...
    六個(gè)周閱讀 8,448評(píng)論 1 33
  • 在react中通過(guò)組件狀態(tài)的改變來(lái)渲染組件樣式 是比較重要模塊罚拟。舉個(gè)例子台诗,一個(gè)組件的顯示形態(tài)是可以由它數(shù)據(jù)狀態(tài)和配...
    夜息白鴿閱讀 114評(píng)論 0 0
  • React 核心思想 —— 組件化React 將界面分成了一個(gè)個(gè)組件,通過(guò)組件的組合赐俗、嵌套構(gòu)成頁(yè)面拉队。其中,組件可復(fù)...
    sylvia_yue閱讀 1,050評(píng)論 1 2
  • 【日精打卡第80天】 姓 名: 夏 婷 公 司:無(wú)錫鎂鈦鉺金屬制品有限公司 六項(xiàng)精進(jìn)第423期 反省二組 【知~學(xué)...
    xia芊芊閱讀 116評(píng)論 0 0