自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

如何解決 React.useEffect() 的無(wú)限循環(huán)

開(kāi)發(fā) 前端
使用useEffect()時(shí),你可能會(huì)遇到一個(gè)陷阱,那就是組件渲染的無(wú)限循環(huán)。在這篇文章中,會(huì)講一下產(chǎn)生無(wú)限循環(huán)的常見(jiàn)場(chǎng)景以及如何避免它們。

[[390825]]

useEffect() 主要用來(lái)管理副作用,比如通過(guò)網(wǎng)絡(luò)抓取、直接操作 DOM、啟動(dòng)和結(jié)束計(jì)時(shí)器。

雖然useEffect() 和 useState(管理狀態(tài)的方法)是最常用的鉤子之一,但需要一些時(shí)間來(lái)熟悉和正確使用。

使用useEffect()時(shí),你可能會(huì)遇到一個(gè)陷阱,那就是組件渲染的無(wú)限循環(huán)。在這篇文章中,會(huì)講一下產(chǎn)生無(wú)限循環(huán)的常見(jiàn)場(chǎng)景以及如何避免它們。

1. 無(wú)限循環(huán)和副作用更新?tīng)顟B(tài)

假設(shè)我們有一個(gè)功能組件,該組件里面有一個(gè) input 元素,組件是功能是計(jì)算 input 更改的次數(shù)。

我們給這個(gè)組件取名為 CountInputChanges,大概的內(nèi)容如下:

  1. function CountInputChanges() { 
  2.   const [value, setValue] = useState(''); 
  3.   const [count, setCount] = useState(-1); 
  4.  
  5.  useEffect(() => setCount(count + 1)); 
  6.   const onChange = ({ target }) => setValue(target.value); 
  7.  
  8.   return ( 
  9.     <div> 
  10.  <input type="text" value={value} onChange={onChange} /> 
  11.  <div>Number of changes: {count}</div> 
  12.  </div> 
  13.   ) 

  <input type =“ text” value = {value} onChange = {onChange} />是受控組件。value變量保存著 input 輸入的值,當(dāng)用戶(hù)輸入輸入時(shí),onChange事件處理程序更新 value 狀態(tài)。

這里使用useEffect()更新count變量。每次由于用戶(hù)輸入而導(dǎo)致組件重新渲染時(shí),useEffect(() => setCount(count + 1))就會(huì)更新計(jì)數(shù)器。

因?yàn)閡seEffect(() => setCount(count + 1))是在沒(méi)有依賴(lài)參數(shù)的情況下使用的,所以()=> setCount(count + 1)會(huì)在每次渲染組件后執(zhí)行回調(diào)。

你覺(jué)得這樣寫(xiě)會(huì)有問(wèn)題嗎?打開(kāi)演示自己試試看:https://codesandbox.io/s/infinite-loop-9rb8c?file=/src/App.js

運(yùn)行了會(huì)發(fā)現(xiàn)count狀態(tài)變量不受控制地增加,即使沒(méi)有在input中輸入任何東西,這是一個(gè)無(wú)限循環(huán)。 å›¾ç‰‡

問(wèn)題在于useEffect()的使用方式:

  1. useEffect(() => setCount(count + 1)); 

它生成一個(gè)無(wú)限循環(huán)的組件重新渲染。

在初始渲染之后,useEffect()執(zhí)行更新?tīng)顟B(tài)的副作用回調(diào)函數(shù)。狀態(tài)更新觸發(fā)重新渲染。重新渲染之后,useEffect()執(zhí)行副作用回調(diào)并再次更新?tīng)顟B(tài),這將再次觸發(fā)重新渲染。


1.1通過(guò)依賴(lài)來(lái)解決

無(wú)限循環(huán)可以通過(guò)正確管理useEffect(callback, dependencies)依賴(lài)項(xiàng)參數(shù)來(lái)修復(fù)。

因?yàn)槲覀兿M鹀ount在值更改時(shí)增加,所以可以簡(jiǎn)單地將value作為副作用的依賴(lài)項(xiàng)。

  1. import { useEffect, useState } from 'react'
  2.  
  3. function CountInputChanges() { 
  4.   const [value, setValue] = useState(''); 
  5.   const [count, setCount] = useState(-1); 
  6.  
  7.  useEffect(() => setCount(count + 1), [value]); 
  8.   const onChange = ({ target }) => setValue(target.value); 
  9.  
  10.   return ( 
  11.     <div> 
  12.  <input type="text" value={value} onChange={onChange} /> 
  13.  <div>Number of changes: {count}</div> 
  14.  </div> 
  15.   ); 

 添加[value]作為useEffect的依賴(lài),這樣只有當(dāng)[value]發(fā)生變化時(shí),計(jì)數(shù)狀態(tài)變量才會(huì)更新。這樣做可以解決無(wú)限循環(huán)。


1.2 使用 ref

除了依賴(lài),我們還可以通過(guò) useRef() 來(lái)解決這個(gè)問(wèn)題。

其思想是更新 Ref 不會(huì)觸發(fā)組件的重新渲染。

  1. import { useEffect, useState, useRef } from "react"
  2.  
  3. function CountInputChanges() { 
  4.   const [value, setValue] = useState(""); 
  5.   const countRef = useRef(0); 
  6.  
  7.  useEffect(() => countRef.current++); 
  8.   const onChange = ({ target }) => setValue(target.value); 
  9.  
  10.   return ( 
  11.     <div> 
  12.  <input type="text" value={value} onChange={onChange} /> 
  13.  <div>Number of changes: {countRef.current}</div> 
  14.  </div> 
  15.   ); 

 useEffect(() => countRef.current++) 每次由于value的變化而重新渲染后,countRef.current++就會(huì)返回。引用更改本身不會(huì)觸發(fā)組件重新渲染。

 

2. 無(wú)限循環(huán)和新對(duì)象引用

即使正確設(shè)置了useEffect()依賴(lài)關(guān)系,使用對(duì)象作為依賴(lài)關(guān)系時(shí)也要小心。

例如,下面的組件CountSecrets監(jiān)聽(tīng)用戶(hù)在input中輸入的單詞,一旦用戶(hù)輸入特殊單詞'secret',統(tǒng)計(jì) 'secret' 的次數(shù)就會(huì)加 1。

  1. import { useEffect, useState } from "react"
  2.  
  3. function CountSecrets() { 
  4.   const [secret, setSecret] = useState({ value: "", countSecrets: 0 }); 
  5.  
  6.   useEffect(() => { 
  7.     if (secret.value === 'secret') { 
  8.  setSecret(s => ({...s, countSecrets: s.countSecrets + 1}));    } 
  9.  }, [secret]); 
  10.   const onChange = ({ target }) => { 
  11.     setSecret(s => ({ ...s, value: target.value })); 
  12.   }; 
  13.  
  14.   return ( 
  15.     <div> 
  16.  <input type="text" value={secret.value} onChange={onChange} /> 
  17.  <div>Number of secrets: {secret.countSecrets}</div> 
  18.  </div> 
  19.   ); 

 打開(kāi)演示(https://codesandbox.io/s/infinite-loop-obj-dependency-7t26v?file=/src/App.js)自己試試,當(dāng)前輸入 secret,secret.countSecrets的值就開(kāi)始不受控制地增長(zhǎng)。

這是一個(gè)無(wú)限循環(huán)問(wèn)題。

為什么會(huì)這樣?

secret對(duì)象被用作useEffect(..., [secret])。在副作用回調(diào)函數(shù)中,只要輸入值等于secret,就會(huì)調(diào)用更新函數(shù)

  1. setSecret(s => ({...s, countSecrets: s.countSecrets + 1})); 

這會(huì)增加countSecrets的值,但也會(huì)創(chuàng)建一個(gè)新對(duì)象。

secret現(xiàn)在是一個(gè)新對(duì)象,依賴(lài)關(guān)系也發(fā)生了變化。所以u(píng)seEffect(..., [secret])再次調(diào)用更新?tīng)顟B(tài)和再次創(chuàng)建新的secret對(duì)象的副作用,以此類(lèi)推。

JavaScript 中的兩個(gè)對(duì)象只有在引用完全相同的對(duì)象時(shí)才相等。

2.1 避免將對(duì)象作為依賴(lài)項(xiàng)

解決由循環(huán)創(chuàng)建新對(duì)象而產(chǎn)生的無(wú)限循環(huán)問(wèn)題的最好方法是避免在useEffect()的dependencies參數(shù)中使用對(duì)象引用。

  1. let count = 0; 
  2.  
  3. useEffect(() => { 
  4.   // some logic 
  5. }, [count]); // Good! 

  1. let myObject = { 
  2.   prop: 'Value' 
  3. }; 
  4.  
  5. useEffect(() => { 
  6.   // some logic 
  7. }, [myObject]); // Not good! 
  8. useEffect(() => { 
  9.   // some logic 
  10. }, [myObject.prop]); // Good! 

修復(fù)組件的無(wú)限循環(huán)問(wèn)題,可以將useEffect(..., [secret])) 變?yōu)?useEffect(..., [secret.value])。

僅在secret.value更改時(shí)調(diào)用副作用回調(diào)就足夠了,下面是修復(fù)后的代碼:

  1. import { useEffect, useState } from "react"
  2.  
  3. function CountSecrets() { 
  4.   const [secret, setSecret] = useState({ value: "", countSecrets: 0 }); 
  5.  
  6.   useEffect(() => { 
  7.     if (secret.value === 'secret') { 
  8.       setSecret(s => ({...s, countSecrets: s.countSecrets + 1})); 
  9.     } 
  10.  }, [secret.value]); 
  11.   const onChange = ({ target }) => { 
  12.     setSecret(s => ({ ...s, value: target.value })); 
  13.   }; 
  14.  
  15.   return ( 
  16.     <div> 
  17.  <input type="text" value={secret.value} onChange={onChange} /> 
  18.  <div>Number of secrets: {secret.countSecrets}</div> 
  19.  </div> 
  20.   ); 

 3 總結(jié)

useEffect(callback, deps)是在組件渲染后執(zhí)行callback(副作用)的 Hook。如果不注意副作用的作用,可能會(huì)觸發(fā)組件渲染的無(wú)限循環(huán)。

生成無(wú)限循環(huán)的常見(jiàn)情況是在副作用中更新?tīng)顟B(tài),沒(méi)有指定任何依賴(lài)參數(shù)

  1. useEffect(() => { 
  2.   // Infinite loop! 
  3.   setState(count + 1); 
  4. }); 

避免無(wú)限循環(huán)的一種有效方法是正確設(shè)置依賴(lài)項(xiàng):

  1. useEffect(() => { 
  2.   // No infinite loop 
  3.   setState(count + 1); 
  4. }, [whenToUpdateValue]); 

另外,也可以使用 Ref,更新 Ref 不會(huì)觸發(fā)重新渲染:

  1. useEffect(() => { 
  2.   // No infinite loop 
  3.   countRef.current++; 
  4. }); 

無(wú)限循環(huán)的另一種常見(jiàn)方法是使用對(duì)象作為useEffect()的依賴(lài)項(xiàng),并在副作用中更新該對(duì)象(有效地創(chuàng)建一個(gè)新對(duì)象)

  1. useEffect(() => { 
  2.   // Infinite loop! 
  3.   setObject({ 
  4.     ...object, 
  5.     prop: 'newValue' 
  6.   }) 
  7. }, [object]); 

避免使用對(duì)象作為依賴(lài)項(xiàng),只使用特定的屬性(最終結(jié)果應(yīng)該是一個(gè)原始值):

  1. useEffect(() => { 
  2.   // No infinite loop 
  3.   setObject({ 
  4.     ...object, 
  5.     prop: 'newValue' 
  6.   }) 
  7. }, [object.whenToUpdateProp]); 

當(dāng)使用useEffect()時(shí),你還知道有其它方式會(huì)引起無(wú)限循環(huán)陷阱嗎?

~完,我是小智,我們下期見(jiàn)~

作者:Shadeed 譯者:前端小智 來(lái)源:dmitripavlutin

原文:https://dmitripavlutin.com/react-useeffect-infinite-loop/

 

責(zé)任編輯:姜華 來(lái)源: 大遷世界
相關(guān)推薦

2010-03-11 14:15:24

Python循環(huán)

2021-09-28 10:32:53

循環(huán)類(lèi)型useEffect

2019-11-26 14:30:20

Spring循環(huán)依賴(lài)Java

2020-12-29 08:34:08

spring循環(huán)依賴(lài)開(kāi)發(fā)

2023-10-07 08:40:57

緩存屬性Spring

2022-08-17 07:52:31

Spring循環(huán)依賴(lài)單例池

2023-11-28 08:00:00

SpringJava

2022-09-19 19:51:30

ReactuseEffect

2023-08-09 10:43:21

源碼循環(huán)依賴(lài)getBean

2015-07-14 10:54:50

PHP數(shù)據(jù)循環(huán)內(nèi)存耗盡

2023-11-26 18:02:00

ReactDOM

2023-12-25 15:40:55

React開(kāi)發(fā)

2020-06-22 08:07:48

Spring依賴(lài)場(chǎng)景

2012-09-05 11:09:15

SELinux操作系統(tǒng)

2017-10-17 09:21:06

2020-11-27 06:28:55

Spring循環(huán)依賴(lài)

2024-12-05 09:06:58

2020-05-19 08:11:09

AI人工智能數(shù)據(jù)

2011-11-03 10:45:09

京東性能瓶頸

2023-07-18 16:05:00

IP地址
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)