React?Hook中的useEffecfa函數(shù)的使用小結(jié)
useEffect的詳細(xì)解析
useEffecf基本使用
書(shū)接上文, 上一篇文章我們講解了State Hook, 我們已經(jīng)可以通過(guò)這個(gè)hook在函數(shù)式組件中定義state
我們知道在類組件中是可以有生命周期函數(shù)的, 那么如何在函數(shù)組件中定義類似于生命周期這些函數(shù)呢?
Effect Hook 可以讓你來(lái)完成一些類似于class中生命周期的功能;
事實(shí)上,類似于網(wǎng)絡(luò)請(qǐng)求、手動(dòng)更新DOM、一些事件的監(jiān)聽(tīng),都是React更新DOM的一些副作用(Side Effects);
所以對(duì)于完成這些功能的Hook被稱之為 Effect Hook;
假如我們現(xiàn)在有一個(gè)需求:頁(yè)面中的title總是顯示counter的數(shù)字,分別使用class組件和Hook實(shí)現(xiàn):
類組件實(shí)現(xiàn)
import React, { PureComponent } from 'react' export class App extends PureComponent { constructor() { super() this.state = { counter: 100 } } // 進(jìn)入頁(yè)面時(shí), 標(biāo)題顯示counter componentDidMount() { document.title = this.state.counter } // 數(shù)據(jù)發(fā)生變化時(shí), 讓標(biāo)題一起變化 componentDidUpdate() { document.title = this.state.counter } render() { const { counter } = this.state return ( <div> <h2>{counter}</h2> <button onClick={() => this.setState({counter: counter+1})}>+1</button> </div> ) } } export default App
函數(shù)組件加Hook的實(shí)現(xiàn):
- 通過(guò)useEffect這個(gè)Hook,可以告訴React需要在渲染后執(zhí)行某些操作;
- useEffect要求我們傳入一個(gè)回調(diào)函數(shù),在React執(zhí)行完更新DOM操作之后(也就是組件被渲染完成后),就會(huì)回調(diào)這個(gè)函數(shù);
- 默認(rèn)情況下,無(wú)論是第一次渲染之后,還是每次更新之后,都會(huì)執(zhí)行這個(gè)回調(diào)函數(shù); 一般情況下我們?cè)谠摶卣{(diào)函數(shù)中都是編寫(xiě)副作用的操作(例如網(wǎng)絡(luò)請(qǐng)求, 操作DOM, 事件監(jiān)聽(tīng))
- 因此需要注意的是, 有許多說(shuō)法說(shuō)useEffect就是用來(lái)模擬生命周期的, 其實(shí)并不是; useEffect可以做到模擬生命周期, 但是他主要的作用是用來(lái)執(zhí)行副作用的
因此需要注意的是, 有許多說(shuō)法說(shuō)useEffect就是用來(lái)模擬生命周期的, 其實(shí)并不是; useEffect可以做到模擬生命周期, 但是他主要的作用是用來(lái)執(zhí)行副作用的
import React, { memo, useEffect, useState } from 'react' const App = memo(() => { const [counter, setCounter] = useState(200) // useEffect傳入一個(gè)回調(diào)函數(shù), 在頁(yè)面渲染完成后自動(dòng)執(zhí)行 useEffect(() => { // 一般在該回調(diào)函數(shù)在編寫(xiě)副作用的代碼(網(wǎng)絡(luò)請(qǐng)求, 操作DOM, 事件監(jiān)聽(tīng)) document.title = counter }) return ( <div> <h2>{counter}</h2> <button onClick={() => setCounter(counter+1)}>+1</button> </div> ) }) export default App
清除副作用(Effect)
在class組件的編寫(xiě)過(guò)程中,某些副作用的代碼,我們需要在componentWillUnmount中進(jìn)行清除:
比如我們之前的事件總線或Redux中手動(dòng)調(diào)用subscribe;
都需要在componentWillUnmount有對(duì)應(yīng)的取消訂閱;
Effect Hook通過(guò)什么方式來(lái)模擬componentWillUnmount呢?
useEffect傳入的回調(diào)函數(shù)A
本身可以有一個(gè)返回值,這個(gè)返回值是另外一個(gè)回調(diào)函數(shù)B
:
type EffectCallback = () => (void | (() => void | undefined));
為什么要在 effect 中返回一個(gè)函數(shù)?
這是 effect 可選的清除機(jī)制。每個(gè) effect 都可以返回一個(gè)清除函數(shù);
如此可以將
添加和移除
訂閱的邏輯放在一起;它們都屬于 effect 的一部分;
React 何時(shí)清除 effect?
React 會(huì)在組件更新和卸載的時(shí)候執(zhí)行清除操作, 將上一次的監(jiān)聽(tīng)取消掉, 只留下當(dāng)前的監(jiān)聽(tīng) ;
正如之前學(xué)到的,effect 在每次渲染的時(shí)候都會(huì)執(zhí)行;
import React, { memo, useEffect } from 'react' const App = memo(() => { useEffect(() => { // 監(jiān)聽(tīng)store數(shù)據(jù)發(fā)生改變 const unsubscribe = store.subscribe(() => { }) // 返回值是一個(gè)回調(diào)函數(shù), 該回調(diào)函數(shù)在組件重新渲染或者要卸載時(shí)執(zhí)行 return () => { // 取消監(jiān)聽(tīng)操作 unsubscribe() } }) return ( <div> <h2>App</h2> </div> ) }) export default App
使用多個(gè)useEffect
使用Hook的其中一個(gè)目的就是解決class中生命周期經(jīng)常將很多的邏輯放在一起的問(wèn)題:
比如網(wǎng)絡(luò)請(qǐng)求、事件監(jiān)聽(tīng)、手動(dòng)修改DOM,這些往往都會(huì)放在componentDidMount中;
一個(gè)函數(shù)組件中可以使用多個(gè)Effect Hook,我們可以將邏輯分離到不同的useEffect中:
import React, { memo, useEffect } from 'react' const App = memo(() => { // 監(jiān)聽(tīng)的useEffect useEffect(() => { console.log("監(jiān)聽(tīng)的代碼邏輯") return () => { console.log("取消的監(jiān)聽(tīng)代碼邏輯") } }) // 發(fā)送網(wǎng)絡(luò)請(qǐng)求的useEffect useEffect(() => { console.log("網(wǎng)絡(luò)請(qǐng)求的代碼邏輯") }) // 操作DOM的useEffect useEffect(() => { console.log("操作DOM的代碼邏輯") }) return ( <div> App </div> ) }) export default App
Hook允許我們按照代碼的用途分離它們, 而不是像生命周期函數(shù)那樣, 將很多邏輯放在一起:
React將按照 effect 聲明的順序
依次調(diào)用
組件中的每一個(gè) effect;
useEffect性能優(yōu)化
默認(rèn)情況下,useEffect的回調(diào)函數(shù)會(huì)在每次渲染時(shí)都重新執(zhí)行,但是這會(huì)導(dǎo)致兩個(gè)問(wèn)題:
某些代碼我們只是希望執(zhí)行一次即可(比如網(wǎng)絡(luò)請(qǐng)求, 組件第一次渲染中執(zhí)行一次即可, 不需要執(zhí)行多次),類似于類組件中的componentDidMount和componentWillUnmount中完成的事情;
另外,多次執(zhí)行也會(huì)導(dǎo)致一定的性能問(wèn)題;
我們?nèi)绾螞Q定useEffect在什么時(shí)候應(yīng)該執(zhí)行和什么時(shí)候不應(yīng)該執(zhí)行呢?
useEffect實(shí)際上有兩個(gè)參數(shù):
- 參數(shù)一: 執(zhí)行的回調(diào)函數(shù), 這個(gè)參數(shù)我們已經(jīng)使用過(guò)了不再多說(shuō);
- 參數(shù)二: 是一個(gè)數(shù)組類型, 表示 該useEffect在哪些state發(fā)生變化時(shí),才重新執(zhí)行;(受誰(shuí)的影響才會(huì)重新執(zhí)行)
案例練習(xí):
受count影響的Effect;
import React, { memo, useEffect, useState } from 'react' const App = memo(() => { const [counter, setCounter] = useState(100) // 發(fā)送網(wǎng)絡(luò)請(qǐng)求的useEffect, 只有在counter發(fā)生改變時(shí)才會(huì)重新執(zhí)行 useEffect(() => { console.log("網(wǎng)絡(luò)請(qǐng)求的代碼邏輯") }, [counter]) return ( <div> <h2 onClick={() => setCounter(counter+1)}>{counter}</h2> </div> ) }) export default App
但是,如果一個(gè)函數(shù)我們不希望依賴任何的內(nèi)容時(shí)
,也可以傳入一個(gè)空的數(shù)組 []:
那么這里的兩個(gè)回調(diào)函數(shù)分別對(duì)應(yīng)的就是componentDidMount和componentWillUnmount生命周期函數(shù)了;
import React, { memo, useEffect, useState } from 'react' const App = memo(() => { const [counter, setCounter] = useState(100) // 傳入空數(shù)組表示不受任何數(shù)據(jù)依賴 useEffect(() => { // 此時(shí)傳入的參數(shù)一這個(gè)回調(diào)函數(shù): 相當(dāng)于componentDidMount console.log("監(jiān)聽(tīng)的代碼邏輯") // 參數(shù)一這個(gè)回調(diào)函數(shù)的返回值: 相當(dāng)于componentWillUnmount return () => { console.log("取消的監(jiān)聽(tīng)代碼邏輯") } }, []) return ( <div> <h2 onClick={() => setCounter(counter+1)}>{counter}</h2> </div> ) }) export default App
總結(jié): useEffect可以模擬之前的class組件的生命周期(類似而不是相等), 并且它比原來(lái)的生命周期更加強(qiáng)大, 青出于藍(lán)而勝于藍(lán)
到此這篇關(guān)于React Hook useEffecfa函數(shù)的使用細(xì)節(jié)詳解的文章就介紹到這了,更多相關(guān)React Hook useEffecfa函數(shù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
React中使用axios發(fā)送請(qǐng)求的幾種常用方法
本文主要介紹了React中使用axios發(fā)送請(qǐng)求的幾種常用方法,主要介紹了get和post請(qǐng)求,具有一定的參考價(jià)值,感興趣的可以了解一下2021-08-08React學(xué)習(xí)之JSX與react事件實(shí)例分析
這篇文章主要介紹了React學(xué)習(xí)之JSX與react事件,結(jié)合實(shí)例形式分析了React中JSX表達(dá)式、屬性、嵌套與react事件相關(guān)使用技巧,需要的朋友可以參考下2020-01-01React-Native實(shí)現(xiàn)ListView組件之上拉刷新實(shí)例(iOS和Android通用)
本篇文章主要介紹了React-Native實(shí)現(xiàn)ListView組件之上拉刷新實(shí)例(iOS和Android通用),具有一定的參考價(jià)值,有興趣的可以了解一下2017-07-07React團(tuán)隊(duì)測(cè)試并發(fā)特性詳解
這篇文章主要為大家介紹了React團(tuán)隊(duì)測(cè)試并發(fā)特性詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-08-08React應(yīng)用框架Dva數(shù)據(jù)流向原理總結(jié)分析
這篇文章主要為大家介紹了React 應(yīng)用框架Dva數(shù)據(jù)流向原理總結(jié)分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-12-12React Hook父組件如何獲取子組件的數(shù)據(jù)/函數(shù)
這篇文章主要介紹了React Hook父組件如何獲取子組件的數(shù)據(jù)/函數(shù),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-09-09