React?useEffect不支持async?function示例分析
引言
useEffect相比大家都耳熟能詳啦,如下這種寫法,應(yīng)該是非常常見的需求。
useEffect(async () => {
await getPoiInfo(); // 請(qǐng)求數(shù)據(jù)
}, []);
但是 React 本身并不支持這么做,理由是 effect function 應(yīng)該返回一個(gè)銷毀函數(shù)(effect:是指return返回的cleanup函數(shù)),如果 useEffect 第一個(gè)參數(shù)傳入 async,返回值則變成了 Promise,會(huì)導(dǎo)致 react 在調(diào)用銷毀函數(shù)的時(shí)候報(bào)錯(cuò) :function.apply is undefined。
React為什么這么設(shè)計(jì)呢?
1、useEffect 的返回值是要在卸載組件時(shí)調(diào)用的,React 需要在 mount 的時(shí)候馬上拿到這個(gè)值,不然就亂套了
2、useEffect() 可能有個(gè)潛在邏輯:第二次觸發(fā) useEffect 里的回調(diào)前,前一次觸發(fā)的行為都執(zhí)行完成,返回的清理函數(shù)也執(zhí)行完成。這樣邏輯才清楚。而如果是異步的,情況會(huì)變得很復(fù)雜,可能會(huì)很容易寫出有 bug 的代碼。
下面有兩種改進(jìn)的方法大家可以參考下:
簡單改造
1、簡單改造的寫法(不推薦)
第一種 在內(nèi)部創(chuàng)建一個(gè)異步函數(shù)anyNameFunction,等待他的結(jié)果,然后調(diào)用setData
但是這種方式存在一個(gè)問題,如果asyncFunction請(qǐng)求有依賴外部的參數(shù),如果不更新requestData 的 effect 的依賴,effect 就不會(huì)同步 props 和 state 帶來的變更,也就不回重新請(qǐng)求數(shù)據(jù)
useEffect(() => {
// Create an scoped async function in the hook
// 注意如果函數(shù)沒有使用組件內(nèi)的任何值,可以把它提到組件外面去定義
// 下面代碼可以提到外面,可以自由地在 effect 中使用,下面就不改啦
async function asyncFunction() {
await requestData();
setData(data)
}
// Execute the created function directly
anyNameFunction();
}, []); // 這里設(shè)置成[]數(shù)組,因?yàn)槲覀冎幌朐趻燧d的時(shí)候運(yùn)行它一次
或者 useEffect中異步函數(shù)采用IIFE寫法( Immediately Invoked Function Expression即立即調(diào)用的函數(shù)式表達(dá)式)
useEffect(() => {
// Using an IIFE
(async function anyNameFunction() {
await requestData();
})();
}, []);
2、把異步提取成單獨(dú)函數(shù)或自定義hook(推薦)
第一種自定義 hook包裹,然后再effect中通過promise.then調(diào)用(github上大佬給的答案:github)
// 自定義hook
function useAsyncEffect(effect: () => Promise<void | (() => void)>, dependencies?: any[]) {
return useEffect(() => {
const cleanupPromise = effect()
return () => { cleanupPromise.then(cleanup => cleanup && cleanup()) }
}, dependencies)
}
// 使用
useAsyncEffect(async () => {
const count = await fetchData()
setCount(count)
}, [fetchData])
或者利用useCallback 包裝成hook
useCallback 本質(zhì)上是添加了一層依賴檢查,使用useCallback,函數(shù)完全可以參與到數(shù)據(jù)流中,可以說如果一個(gè)函數(shù)的輸入改變了,這個(gè)函數(shù)就改變了,如果沒有,函數(shù)也不會(huì)改變。
下面的例子中會(huì)依賴 type ,如果 type 保持不變,requestData 也會(huì)保持不變,effect 也不會(huì)重新運(yùn)行,但是如果 type 修改了,requestData 也會(huì)隨之改變,因此會(huì)重新請(qǐng)求數(shù)據(jù)。
// 封裝
const requestData = useCallback(async () => {
changeLoading(true);
changeError(false);
changeList([]);
requestAPI.getFeature({ type }).then((data) => {
if (data) {
changeList(data);
}
}).catch((e) => {
changeError(true);
}).finally(() => {
changeLoading(false);
});
}, [type]); // type改變會(huì)重新生成函數(shù)
// 普通接口請(qǐng)求
useEffect(() => {
requestData();
}, [requestData]);
// 單獨(dú)處理外層刷新的接口請(qǐng)求
// refreshing是props傳遞的過來的,不應(yīng)該與state狀態(tài)改變混在一起,這也是hook的優(yōu)勢,將不相關(guān)的狀態(tài)邏輯拆分成更細(xì)粒度
useEffect(() => {
if (!refreshing) {
return;
}
requestData().then(() => {
getRefreshStatus(false);
});
}, [refreshing]);
關(guān)于為什么不支持異步的原理可以看下這篇文章里通過源碼的分析:useEffect 中為啥不能使用 async
有任何疑問歡迎評(píng)論溝通,我會(huì)繼續(xù)更新!
其他相關(guān)文檔:
https://heptaluan.github.io/2020/11/07/React/17/
https://www.robinwieruch.de/react-hooks-fetch-data/
以上就是React useEffect不支持async function示例分析的詳細(xì)內(nèi)容,更多關(guān)于useEffect不支持async function的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
React替換傳統(tǒng)拷貝方法的Immutable使用
Immutable.js出自Facebook,是最流行的不可變數(shù)據(jù)結(jié)構(gòu)的實(shí)現(xiàn)之一。它實(shí)現(xiàn)了完全的持久化數(shù)據(jù)結(jié)構(gòu),使用結(jié)構(gòu)共享。所有的更新操作都會(huì)返回新的值,但是在內(nèi)部結(jié)構(gòu)是共享的,來減少內(nèi)存占用2023-02-02
React服務(wù)端渲染和同構(gòu)的實(shí)現(xiàn)
本文主要介紹了React服務(wù)端渲染和同構(gòu)的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-04-04
react項(xiàng)目自行配置熱更新的實(shí)現(xiàn)
本文主要介紹了react項(xiàng)目自行配置熱更新的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-11-11
jsoneditor二次封裝實(shí)時(shí)預(yù)覽json編輯器組件react版
這篇文章主要為大家介紹了jsoneditor二次封裝實(shí)時(shí)預(yù)覽json編輯器組件react版示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-10-10

