溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

react中如何使用useEffect

發(fā)布時間:2022-08-08 15:05:02 來源:億速云 閱讀:153 作者:iii 欄目:開發(fā)技術(shù)

這篇文章主要講解了“react中如何使用useEffect”,文中的講解內(nèi)容簡單清晰,易于學(xué)習(xí)與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學(xué)習(xí)“react中如何使用useEffect”吧!

使用useEffect及踩坑記錄

useEffect 介紹

useEffect時reactHook中最重要,最常用的hook之一。

useEffect相當(dāng)于react中的什么生命周期呢?

這個問題在react官網(wǎng)中有過介紹,在使用的過程中,容易被忽略,在面試的時候卻經(jīng)常被問及,(面試造航母,上班擰螺絲?),開個玩笑這個問題并不難回答,下面是react官方的原話:

如果你熟悉 React class 的生命周期函數(shù),你可以把 useEffect Hook 看做 componentDidMount,componentDidUpdate 和 componentWillUnmount 這三個函數(shù)的組合。

  • componentDidMount 組件掛載

  • componentDidUpdate 組件更新

  • componentWillUnmount 組件將要摧毀

useEffect需要傳遞兩個參數(shù),第一個參數(shù)是邏輯處理函數(shù),第二個參數(shù)是一個數(shù)組

用法

useEffect(() => {
/** 執(zhí)行邏輯 */
},[])

重要理解

一、第二個參數(shù)存放變量,當(dāng)數(shù)組存放變量發(fā)生改變時,第一個參數(shù),邏輯處理函數(shù)將會被執(zhí)行

二、第二個參數(shù)可以不傳,不會報錯,但瀏覽器會無線循環(huán)執(zhí)行邏輯處理函數(shù)。

useEffect(() => {
/** 執(zhí)行邏輯 */
})

三、第二個參數(shù)如果只傳一個空數(shù)組,邏輯處理函數(shù)里面的邏輯只會在組件掛載時執(zhí)行一次 ,不就是相當(dāng)于 componentDidMount

useEffect(() => {
/** 執(zhí)行邏輯 */
},[])

四、第二個參數(shù)如果不為空數(shù)組,如下

const [a, setA] = useState(1);
const [b, setB] = useState(2);
useEffect(() => {
/** 執(zhí)行邏輯 */
},[a,b])

邏輯處理函數(shù)會在組件掛載時執(zhí)行一次和(a或者b變量在棧中的值發(fā)生改變時執(zhí)行一次) 這是不是相當(dāng)于componentDidMount 和 componentDidUpdate 的結(jié)合

五、useEffect第一個參數(shù)可以返回一個回調(diào)函數(shù),這個回調(diào)函數(shù)將會在組件被摧毀之前和再一次觸發(fā)更新時,將之前的副作用清除掉。這就相當(dāng)于componentWillUnmount。

useEffect去除副作用。我們可能會在組件即將被掛載的時候創(chuàng)建一些不斷循環(huán)的訂閱(計時器,或者遞歸循環(huán))。在組件被摧毀之前,或者依賴數(shù)組的元素更新后,應(yīng)該將這些訂閱也給摧毀掉。

比如以下的情況(沒有去除計時器,增大不必要的開銷和代碼風(fēng)險)

const [time, setTime] = useState(0)
useEffect(() => {
    const InterVal = setInterval(() => {
        setTime(time + 1)
    },1000)
},[])

利用第五點,在組件被摧毀前去除計時器。

const [time, setTime] = useState(0)
useEffect(() => {
    const InterVal = setInterval(() => {
        setTime(time + 1)
    },1000)
    return () => {
           clearInterval(InterVal )
       }
},[])

useEffect常見跳坑

1、useEffect執(zhí)行函數(shù)被循環(huán)執(zhí)行。

出現(xiàn)這種情況可能有兩種原因。

沒傳第二個參數(shù)

useEffect(() => {
/** 執(zhí)行邏輯 */
})

2、你在useEffect執(zhí)行函數(shù)里面改變了useEffect監(jiān)測的變量

const [a, setA] = useState(1);
useEffect(() => {
/** 執(zhí)行邏輯 */
setA(a + 1)
},[a])

解決的方法 不要在useEffect第一參數(shù)執(zhí)行函數(shù)中去改變第二參數(shù)依賴元素的地址的值。當(dāng)依賴元素的地址的值發(fā)生改變,又會執(zhí)行一次執(zhí)行函數(shù),這不是無限循環(huán)么。

3、useEffect監(jiān)測不到依賴數(shù)組元素的變化。

只有一種可能,依賴數(shù)組元素的地址的值根本就沒變,比如:

const [a, setA] = useState({
b: 'dx',
c: '18',
})
const changeA = () => {
    setA((old) => {
    old.b = 'yx'
    return old
    })
}
useEffect(() => {
/** 當(dāng)組件掛載時執(zhí)行一次changeA */
changeA ()
},[])
/**當(dāng)changeA執(zhí)行卻沒有打印 a*/
useEffect(() => {
/** 執(zhí)行邏輯 */
console.log(a)
},[a])

是因為changeA沒有真正的改變a在棧中的值(地址的值),只是改變了a在堆中的值。

useEffect監(jiān)測不到堆中值得變化,所有引用類型數(shù)據(jù)都應(yīng)該注意這一點。

解決的辦法:

const [a, setA] = useState({
b: 'dx',
c: '18',
})
const changeA = () => {
    setA((old) => {
    const newA = {...old}
    newA .b = 'yx'
    return newA 
    })
}
useEffect(() => {
/** 當(dāng)組件掛載時執(zhí)行一次changeA */
changeA ()
},[])
/**當(dāng)changeA執(zhí)行打印  {b:'yx',c:'18'}  */
useEffect(() => {
/** 執(zhí)行邏輯 */
console.log(a)
},[a])

hooks中useEffect()使用總結(jié)

常見使用

獲取數(shù)據(jù)案例:

import React, { useState, useEffect } from 'react';
import axios from 'axios';
function App() {
  const [data, setData] = useState({ hits: [] });
  useEffect(() => {
    const fetchData = async () => {
      const result = await axios(
        'https://hn.algolia.com/api/v1/search?query=redux',
      );
      setData(result.data);
    };
    fetchData();
  }, []);
  return (
    <ul>
      {data.hits.map(item => (
        <li key={item.objectID}>
          <a href={item.url}>{item.title}</a>
        </li>
      ))}
    </ul>
  );
}
export default App;

上面例子中,useState()用來生成一個狀態(tài)變量(data),保存獲取的數(shù)據(jù);useEffect()的副效應(yīng)函數(shù)內(nèi)部有一個 async 函數(shù),用來從服務(wù)器異步獲取數(shù)據(jù)。拿到數(shù)據(jù)以后,再用setData()觸發(fā)組件的重新渲染。

由于獲取數(shù)據(jù)只需要執(zhí)行一次,所以上例的useEffect()的第二個參數(shù)為一個空數(shù)組

附線上運行代碼

useEffect() 的第二個參數(shù)說明

有時候,我們不希望useEffect()每次渲染都執(zhí)行,這時可以使用它的第二個參數(shù),使用一個數(shù)組指定副效應(yīng)函數(shù)的依賴項,只有依賴項發(fā)生變化,才會重新渲染。

function Welcome(props) {
  useEffect(() => {
    document.title = `Hello, ${props.name}`;
  }, [props.name]);
  return <h2>Hello, {props.name}</h2>;
}

上面例子中,useEffect()的第二個參數(shù)是一個數(shù)組,指定了第一個參數(shù)(副效應(yīng)函數(shù))的依賴項(props.name)。只有該變量發(fā)生變化時,副效應(yīng)函數(shù)才會執(zhí)行。如果第二個參數(shù)是一個空數(shù)組,就表明副效應(yīng)參數(shù)沒有任何依賴項。

因此,副效應(yīng)函數(shù)這時只會在組件加載進入 DOM 后執(zhí)行一次,后面組件重新渲染,就不會再次執(zhí)行。這很合理,由于副效應(yīng)不依賴任何變量,所以那些變量無論怎么變,副效應(yīng)函數(shù)的執(zhí)行結(jié)果都不會改變,所以運行一次就夠了。

useEffect() 第一個函數(shù)參數(shù)的返回值

副效應(yīng)是隨著組件加載而發(fā)生的,那么組件卸載時,可能需要清理這些副效應(yīng)。

useEffect()允許返回一個函數(shù),在組件卸載時,執(zhí)行該函數(shù),清理副效應(yīng)。如果不需要清理副效應(yīng),useEffect()就不用返回任何值。

useEffect(() => {
  const subscription = props.source.subscribe();
  return () => {
    subscription.unsubscribe();
  };
}, [props.source]);

上面例子中,useEffect()在組件加載時訂閱了一個事件,并且返回一個清理函數(shù),在組件卸載時取消訂閱。

實際使用中,由于副效應(yīng)函數(shù)默認(rèn)是每次渲染都會執(zhí)行,所以清理函數(shù)不僅會在組件卸載時執(zhí)行一次,每次副效應(yīng)函數(shù)重新執(zhí)行之前,也會執(zhí)行一次,用來清理上一次渲染的副效應(yīng)。

useEffect() 的注意點

使用useEffect()時,有一點需要注意。如果有多個副效應(yīng),應(yīng)該調(diào)用多個useEffect(),而不應(yīng)該合并寫在一起。

錯誤寫法:

function App() {
  const [varA, setVarA] = useState(0);
  const [varB, setVarB] = useState(0);
  useEffect(() => {
    const timeoutA = setTimeout(() => setVarA(varA + 1), 1000);
    const timeoutB = setTimeout(() => setVarB(varB + 2), 2000);
    return () => {
      clearTimeout(timeoutA);
      clearTimeout(timeoutB);
    };
  }, [varA, varB]);
  return <span>{varA}, {varB}</span>;
}

上面的例子是錯誤的寫法,副效應(yīng)函數(shù)里面有兩個定時器,它們之間并沒有關(guān)系,其實是兩個不相關(guān)的副效應(yīng),不應(yīng)該寫在一起。正確的寫法是將它們分開寫成兩個useEffect()。

正確寫法:

function App() {
  const [varA, setVarA] = useState(0);
  const [varB, setVarB] = useState(0);
  useEffect(() => {
    const timeout = setTimeout(() => setVarA(varA + 1), 1000);
    return () => clearTimeout(timeout);
  }, [varA]);
  useEffect(() => {
    const timeout = setTimeout(() => setVarB(varB + 2), 2000);
    return () => clearTimeout(timeout);
  }, [varB]);
  return <span>{varA}, {varB}</span>;
}

感謝各位的閱讀,以上就是“react中如何使用useEffect”的內(nèi)容了,經(jīng)過本文的學(xué)習(xí)后,相信大家對react中如何使用useEffect這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關(guān)知識點的文章,歡迎關(guān)注!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI