淺析React Hooks

在react最新的一個(gè)大版本中曲初,正式的給我們帶來(lái)了hook:React v16.8: The One With Hooks就是讓我們不再只能使用 class 來(lái)實(shí)現(xiàn)有狀態(tài)的組件管毙。

熟悉react的朋友應(yīng)該都了解铃将,當(dāng)我們需要具有狀態(tài)的組件食店,或者需要使用生命周期函數(shù)來(lái)完成一些操作的時(shí)候霎奢,就不可避免的需要這樣:

class Component extend React.Component {
    ...
}

如果這個(gè)組件內(nèi)部足夠復(fù)雜那倒還行蠕蚜,但如果只是涉及簡(jiǎn)單的數(shù)據(jù)改變,而讓我們每次實(shí)現(xiàn)它都需要實(shí)現(xiàn)一個(gè)class這也是一個(gè)讓人很惱火的事情乎折,同時(shí)也增加了我們?nèi)粘i_(kāi)發(fā)的代碼量绒疗。但有了hook之后,我們就可以愉快的使用hook來(lái)實(shí)現(xiàn)由狀態(tài)的函數(shù)組件了骂澄。

在hook里面主要有以下兩個(gè)比較常用的API:

  • useState
  • useEffect

useState

useState所要解決的問(wèn)題就是在函數(shù)中使用狀態(tài)吓蘑。比如我們需要實(shí)現(xiàn)一個(gè)可以加減的計(jì)數(shù)器:

import { useState } from 'react'

const Counter = () => {
  const [count, setCount] = useState(0)

  return (
    <div>
       <div>{count}</div>
       <button onClick={() => setCount(count + 1)}>+</button>
       <button onClick={() => setCount(count - 1)}>-</button>
    </div>
  );
};

在這里我們就是使用useState來(lái)在Counter函數(shù)內(nèi)部創(chuàng)建狀態(tài),并管理狀態(tài)坟冲。

useState 接受一個(gè)參數(shù)磨镶,也就是 state 的初始值,它的返回值是一個(gè)只有兩個(gè)元素的數(shù)組健提,其中第一個(gè)元素是 state 的值琳猫,第二個(gè)元素是用于更新 state 的函數(shù)。上面那段代碼就是使用了ES6中的解構(gòu)賦值來(lái)把兩個(gè)元素分別賦值給 count 和 setCount私痹。當(dāng)然脐嫂,count和setCount 這兩個(gè)名字并不是固定的,我們可以給這兩個(gè)元素隨意的設(shè)置名字紊遵。

useState在處理簡(jiǎn)單的數(shù)據(jù)流時(shí)相較之使用class要簡(jiǎn)明方便的多账千,下面是在結(jié)合antd的代碼可以感受一下代碼量上的差距:

class Modal extends React.Component {
  state = { visible: false }

  showModal = () => {
    this.setState({
      visible: true,
    });
  }
  handleOk = (e) => {
    console.log(e);
    this.setState({
      visible: false,
    });
  }
  handleCancel = (e) => {
    console.log(e);
    this.setState({
      visible: false,
    });
  }
  render() {
    return (
      <div>
        <Button type="primary" onClick={this.showModal}>
          Open Modal
        </Button>
        <Modal
          title="Basic Modal"
          visible={this.state.visible}
          onOk={this.handleOk}
          onCancel={this.handleCancel}
        >
          <p>this is a modal</p>
        </Modal>
      </div>
    );
  }
}

這是antd所提供的一個(gè)關(guān)于model的組件,其內(nèi)部就是通過(guò)維護(hù)visible的狀態(tài)來(lái)控制這個(gè)model是否顯示暗膜。我們可以看到匀奏,雖然只是簡(jiǎn)單的兩種狀態(tài),但我們?cè)趧?chuàng)建visible這一state的同時(shí)桦山,還需要使用三個(gè)函數(shù)去控制改變它。而當(dāng)我們使用useState后就很簡(jiǎn)單了:

const Modal = () => {
  const [visible , changeVisible] = useState(false)
  return (
    <div>
      <Button type="primary" onClick={()=>changeVisible(true)}>open</Button>
      <Modal
          title="Basic Modal"
          visible={visible}
          onOk={()=>changeVisible(false)}
          onCancel={()=>changeVisible(false)}
        >
          <p>this is a modal</p>
        </Modal>
    </div>
  )
}

useEffect

上面提到醋旦,在需要使用狀態(tài)和生命周期函數(shù)的時(shí)候我們需要使用class恒水,上面的useState是為我們解決狀態(tài)問(wèn)題的,而這個(gè)userEffect則是為我們解決在函數(shù)組件中生命周期的問(wèn)題的饲齐。

在 Counter 組件中钉凌,如果我們想要在用戶點(diǎn)擊“+”或者“-”按鈕之后把計(jì)數(shù)值體現(xiàn)在網(wǎng)頁(yè)標(biāo)題上,這就是一個(gè)修改 DOM 的副作用操作捂人,所以我們需要使用componentDidMount或者componentDidUpdate御雕,因此我們得把 Counter 寫(xiě)成 class 組件矢沿,但在在使用useEffect后我們可以這么寫(xiě):

import { useState, useEffect } from 'react'

const Counter = () => {
  const [count, setCount] = useState(0)
  
  useEffect(() => {
    document.title = `Count: ${count}`
  })

  return (
    <div>
       <div>{count}</div>
       <button onClick={() => setCount(count + 1)}>+</button>
       <button onClick={() => setCount(count - 1)}>-</button>
    </div>
  )
}

在這里userEffect就接受一個(gè)函數(shù)作為參數(shù),每次Counter組件發(fā)生渲染的時(shí)候酸纲,都會(huì)調(diào)用這個(gè)函數(shù)捣鲸,也就做到了上面所說(shuō)的那兩個(gè)生命周期函數(shù)的作用。

但需要注意的是闽坡,如果我們只傳入一個(gè)參數(shù)的話栽惶,useEffect會(huì)在每次渲染過(guò)后都調(diào)用這個(gè)函數(shù),而在我們?nèi)粘i_(kāi)發(fā)使用componentDidMount的時(shí)候疾嗅,往往可能只需要當(dāng)mount后才調(diào)用外厂,比如當(dāng)我們需要在掛載組件后去請(qǐng)求某個(gè)數(shù)據(jù),但當(dāng)update的時(shí)候則不需要去請(qǐng)求它代承。在useEffect中同樣也有方法可以實(shí)現(xiàn)這種操作汁蝶,useEffect 其實(shí)還有第二個(gè)可選參數(shù),只有當(dāng)同一 useEffect 的兩次調(diào)用的第二個(gè)參數(shù)不同時(shí)论悴,第一個(gè)參數(shù)才會(huì)被調(diào)用掖棉,所以相實(shí)現(xiàn)上面這種功能,我們只需要這樣寫(xiě):

useEffect(() => {
    console.log("在這里只有第一次mount的時(shí)候才會(huì)被執(zhí)行")
}, [666]);

還有一點(diǎn)需要注意的是意荤,所有的 Hooks API 都只能在函數(shù)類型組件中使用啊片,而class 類型的組件是不能使用的。(不過(guò)也對(duì)hook本身誕生的目的就是為函數(shù)式組件服務(wù)的)

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末玖像,一起剝皮案震驚了整個(gè)濱河市紫谷,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌捐寥,老刑警劉巖笤昨,帶你破解...
    沈念sama閱讀 206,378評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異握恳,居然都是意外死亡瞒窒,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,356評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門乡洼,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)崇裁,“玉大人,你說(shuō)我怎么就攤上這事束昵“挝龋” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 152,702評(píng)論 0 342
  • 文/不壞的土叔 我叫張陵锹雏,是天一觀的道長(zhǎng)巴比。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么轻绞? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 55,259評(píng)論 1 279
  • 正文 為了忘掉前任采记,我火速辦了婚禮,結(jié)果婚禮上政勃,老公的妹妹穿的比我還像新娘唧龄。我一直安慰自己,他們只是感情好稼病,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,263評(píng)論 5 371
  • 文/花漫 我一把揭開(kāi)白布选侨。 她就那樣靜靜地躺著,像睡著了一般然走。 火紅的嫁衣襯著肌膚如雪援制。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 49,036評(píng)論 1 285
  • 那天芍瑞,我揣著相機(jī)與錄音晨仑,去河邊找鬼。 笑死拆檬,一個(gè)胖子當(dāng)著我的面吹牛洪己,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播竟贯,決...
    沈念sama閱讀 38,349評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼答捕,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了屑那?” 一聲冷哼從身側(cè)響起拱镐,我...
    開(kāi)封第一講書(shū)人閱讀 36,979評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎持际,沒(méi)想到半個(gè)月后沃琅,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,469評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡蜘欲,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,938評(píng)論 2 323
  • 正文 我和宋清朗相戀三年益眉,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片姥份。...
    茶點(diǎn)故事閱讀 38,059評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡郭脂,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出澈歉,到底是詐尸還是另有隱情展鸡,我是刑警寧澤,帶...
    沈念sama閱讀 33,703評(píng)論 4 323
  • 正文 年R本政府宣布闷祥,位于F島的核電站娱颊,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏凯砍。R本人自食惡果不足惜箱硕,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,257評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望悟衩。 院中可真熱鬧剧罩,春花似錦、人聲如沸座泳。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,262評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)挑势。三九已至镇防,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間潮饱,已是汗流浹背来氧。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,485評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留香拉,地道東北人啦扬。 一個(gè)月前我還...
    沈念sama閱讀 45,501評(píng)論 2 354
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像凫碌,于是被迫代替她去往敵國(guó)和親扑毡。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,792評(píng)論 2 345