成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

使用React Hooks 時要避免的5個錯誤!

開發 前端
很有可能你已經讀過很多關于如何使用React Hook 的文章。但有時候,知道何時不使用與知道如何使用同樣重要。

[[386276]]

本文已經過原作者 Shadeed 授權翻譯。

很有可能你已經讀過很多關于如何使用React Hook 的文章。但有時候,知道何時不使用與知道如何使用同樣重要。

在這篇文章中,主要介紹一下 React hooks 錯誤使用方式,以及如何解決它們。

  • 不要更改 Hook 調用順序
  • 不要使用過時狀態
  • 不要創建過時的閉包
  • 不要將狀態用于基礎結構數據
  • 不要忘記清理副作用

1.不要更改 Hook 調用順序

在寫這篇文章的前幾天,我編寫了一個通過id獲取游戲信息的組件,下面是一個簡單的版本 FetchGame:

  1. function FetchGame({ id }) { 
  2.   if (!id) { 
  3.     return 'Please select a game to fetch'
  4.   } 
  5.  
  6.   const [game, setGame] = useState({  
  7.     name''
  8.     description: ''  
  9.   }); 
  10.  
  11.   useEffect(() => { 
  12.     const fetchGame = async () => { 
  13.       const response = await fetch(`/api/game/${id}`); 
  14.       const fetchedGame = await response.json(); 
  15.       setGame(fetchedGame); 
  16.     }; 
  17.     fetchGame(); 
  18.   }, [id]); 
  19.  
  20.   return ( 
  21.     <div> <div>Name: {game.name}</div> <div>Description: {game.description}</div> </div> 
  22.   );  

 

組件FetchGame 接收 id(即要獲取的游戲的ID)。useEffect() 在await fetch(/game/${id})提取游戲信息并將其保存到狀態變量game中。

打開演示(https://codesandbox.io/s/hooks-order-warning-rdxpg?file=/pages/index.js) 。組件正確地執行獲取操作,并使用獲取的數據更新狀態。但是看看tab Eslint警告: 有 Hook 執行順序不正確的問題。

 

問題發生在這一判斷:

  1. function FetchGame({ id }) { 
  2.  if (!id) { return 'Please select a game to fetch'; }   
  3.    // ... 

當id為空時,組件渲染'Please select a game to fetch'并退出,不調用任何 Hook。

但是,如果 id不為空(例如等于'1'),則會調用useState()和 useEffect()。

有條件地執行 Hook 可能會導致難以調試的意外錯誤。React Hook的內部工作方式要求組件在渲染之間總是以相同的順序調用 Hook。

這正是鉤子的第一條規則:不要在循環、條件或嵌套函數內調用 Hook。

解決方法就是將條件判斷放到 Hook 后面:

  1. function FetchGame({ id }) { 
  2.   const [game, setGame] = useState({  
  3.     name''
  4.     description: ''  
  5.   }); 
  6.  
  7.   useEffect(() => { 
  8.     const fetchGame = async () => { 
  9.       const response = await fetch(`/api/game/${id}`); 
  10.       const fetchedGame = await response.json(); 
  11.       setGame(fetchedGame); 
  12.     }; 
  13.  if (id) {      fetchGame();     }  }, [id]); 
  14.  
  15.  if (!id) { return 'Please select a game to fetch'; } 
  16.   return ( 
  17.     <div> 
  18.  <div>Name: {game.name}</div> 
  19.  <div>Description: {game.description}</div> 
  20.  </div> 
  21.   ); 

 

現在,無論id是否為空,useState()和useEffect() 總是以相同的順序被調用,這就是 Hook 應該始終被調用的方式。

2.不要使用過時狀態

下面的組件MyIncreaser在單擊按鈕時增加狀態變量count:

  1. function MyIncreaser() { 
  2.   const [count, setCount] = useState(0); 
  3.  
  4.   const increase = useCallback(() => { 
  5.     setCount(count + 1); 
  6.   }, [count]); 
  7.  
  8.   const handleClick = () { 
  9.  increase(); increase(); increase();  }; 
  10.  
  11.   return ( 
  12.     <> 
  13.  <button onClick={handleClick}>Increase</button> 
  14.  <div>Counter: {count}</div> 
  15.  </> 
  16.   ); 

 

這里有趣一點的是,handleClick調用了3次狀態更新。

現在,在打開演示之前,問一個問題:如果單擊一次按鈕,計數器是否增加3?

打開演示(https://codesandbox.io/s/stale-variable-jo32q?file=/src/index.js),點擊按鈕一次,看看結果。

不好意思,即使在handleClick()中3次調用了increase(),計數也只增加了1。

問題在于setCount(count + 1)狀態更新器。當按鈕被點擊時,React調用setCount(count + 1)3次

  1. const handleClick = () { 
  2.    increase(); 
  3.    increase(); 
  4.    increase(); 
  5.  }; 
  6.  
  7. / 等價: 
  8.  
  9.  const handleClick = () { 
  10.    setCount(count + 1); 
  11.    // count variable is now stale 
  12.    setCount(count + 1); 
  13.    setCount(count + 1); 
  14.  }; 

setCount(count + 1)的第一次調用正確地將計數器更新為count + 1 = 0 + 1 = 1。但是,接下來的兩次setCount(count + 1)調用也將計數設置為1,因為它們使用了過時的stale狀態。

通過使用函數方式更新狀態來解決過時的狀態。我們用setCount(count => count + 1)代替setCount(count + 1):

  1. function MyIncreaser() { 
  2.   const [count, setCount] = useState(0); 
  3.  
  4.   const increase = useCallback(() => { 
  5.  setCount(count => count + 1);  }, []); 
  6.  
  7.   const handleClick = () { 
  8.     increase(); 
  9.     increase(); 
  10.     increase(); 
  11.   }; 
  12.  
  13.   return ( 
  14.     <> 
  15.  <button onClick={handleClick}>Increase</button> 
  16.  <div>Counter: {count}</div> 
  17.  </> 
  18.   ); 

 

這里有一個好規則可以避免遇到過時的變量:

如果你使用當前狀態來計算下一個狀態,總是使用函數方式來更新狀態:setValue(prevValue => prevValue + someResult)。

3.不要創建過時的閉包

React Hook 很大程序上依賴于閉包的概念。依賴閉包是它們如此富有表現力的原因。

JavaScript 中的閉包是從其詞法作用域捕獲變量的函數。不管閉包在哪里執行,它總是可以從定義它的地方訪問變量。

當使用 Hook 接受回調作為參數時(如useEffect(callback, deps), useCallback(callback, deps)),你可能會創建一個過時的閉包,一個捕獲了過時的狀態或變量的閉包。

我們來看看一個使用useEffect(callback, deps) 而忘記正確設置依賴關系時創建的過時閉包的例子。

在組件中,useEffect()每2秒打印一次count的值

  1.  const [count, setCount] = useState(0); 
  2.  
  3.   useEffect(function() { 
  4.     setInterval(function log() { 
  5.       console.log(`Count is: ${count}`); 
  6.     }, 2000); 
  7.   }, []); 
  8.  
  9.   const handleClick = () => setCount(count => count + 1); 
  10.  
  11.   return ( 
  12.     <> <button onClick={handleClick}>Increase</button> <div>Counter: {count}</div> </> 
  13.   ); 

 

打開演示(https://codesandbox.io/s/stale-variable-jo32q?file=/src/index.js),點擊按鈕。在控制臺查看,每2秒打印的都 是 Count is: 0,,不管count狀態變量的實際值是多少。

為啥這樣子?

第一次渲染時, log 函數捕獲到的 count 的值為 0。

之后,當按鈕被單擊并且count增加時,setInterval取到的 count 值仍然是從初始渲染中捕獲count為0的值。log 函數是一個過時的閉包,因為它捕獲了一個過時的狀態變量count。

解決方案是讓useEffect()知道閉包log依賴于count,并正確重置計時器

  1. function WatchCount() { 
  2.   const [count, setCount] = useState(0); 
  3.  
  4.   useEffect(function() { 
  5.     const id = setInterval(function log() { 
  6.       console.log(`Count is: ${count}`); 
  7.     }, 2000); 
  8.  return () => clearInterval(id); }, [count]); 
  9.   const handleClick = () => setCount(count => count + 1); 
  10.  
  11.   return ( 
  12.     <> 
  13.  <button onClick={handleClick}>Increase</button> 
  14.  <div>Counter: {count}</div> 
  15.  </> 
  16.   ); 

 

正確設置依賴關系后,一旦count發生變化,useEffect()就會更新setInterval()的閉包。

為了防止閉包捕獲舊值:確保提供給 Hook 的回調函數中使用依賴項。

4.不要將狀態用于基礎結構數據

有一次,我需要在狀態更新上調用副作用,在第一個渲染不用調用副作用。useEffect(callback, deps)總是在掛載組件后調用回調函數:所以我想避免這種情況。

我找到了以下的解決方案

  1. function MyComponent() { 
  2.   const [isFirst, setIsFirst] = useState(true); 
  3.   const [count, setCount] = useState(0); 
  4.  
  5.   useEffect(() => { 
  6.     if (isFirst) { 
  7.       setIsFirst(false); 
  8.       return
  9.     } 
  10.     console.log('The counter increased!'); 
  11.   }, [count]); 
  12.  
  13.   return ( 
  14.     <button onClick={() => setCount(count => count + 1)}> Increase </button> 
  15.   ); 

狀態變量isFirst用來判斷是否是第一次渲染。一旦更新setIsFirst(false),就會出現另一個無緣無故的重新渲染。

保持count狀態是有意義的,因為界面需要渲染 count 的值。但是,isFirst不能直接用于計算輸出。

是否為第一個渲染的信息不應存儲在該狀態中。基礎結構數據,例如有關渲染周期(即首次渲染,渲染數量),計時器ID(setTimeout(),setInterval()),對DOM元素的直接引用等詳細信息,應使用引用useRef()進行存儲和更新。

我們將有關首次渲染的信息存儲到 Ref 中:

  1. const isFirstRef = useRef(true);  const [count, setCount] = useState(0); 
  2.  
  3.  useEffect(() => { 
  4.    if (isFirstRef.current) { 
  5.      isFirstRef.current = false
  6.      return
  7.    } 
  8.    console.log('The counter increased!'); 
  9.  }, [count]); 
  10.  
  11.  return ( 
  12.    <button onClick={() => setCounter(count => count + 1)}> 
  13. Increase 
  14. </button> 
  15.  ); 

isFirstRef是一個引用,用于保存是否為組件的第一個渲染的信息。isFirstRef.current屬性用于訪問和更新引用的值。

重要說明:更新參考isFirstRef.current = false不會觸發重新渲染。

5.不要忘記清理副作用

很多副作用,比如獲取請求或使用setTimeout()這樣的計時器,都是異步的。

如果組件卸載或不再需要該副作用的結果,請不要忘記清理該副作用。

下面的組件有一個按鈕。當按鈕被點擊時,計數器每秒鐘延遲增加1:

  1. function DelayedIncreaser() { 
  2.   const [count, setCount] = useState(0); 
  3.   const [increase, setShouldIncrease] = useState(false); 
  4.  
  5.   useEffect(() => { 
  6.     if (increase) { 
  7.       setInterval(() => { 
  8.         setCount(count => count + 1) 
  9.       }, 1000); 
  10.     } 
  11.   }, [increase]); 
  12.  
  13.   return ( 
  14.     <> <button onClick={() => setShouldIncrease(true)}> Start increasing </button> <div>Count: {count}</div> </> 
  15.   ); 

 

打開演示(https://codesandbox.io/s/unmounted-state-update-n1d3u?file=/src/index.js),點擊開始按鈕。正如預期的那樣,狀態變量count每秒鐘都會增加。

在進行遞增操作時,單擊umount 按鈕,卸載組件。React會在控制臺中警告更新卸載組件的狀態。

 

修復DelayedIncreaser很簡單:只需從useEffect()的回調中返回清除函數:

  1. // ... 
  2.  
  3.  useEffect(() => { 
  4.    if (increase) { 
  5.      const id = setInterval(() => { 
  6.        setCount(count => count + 1) 
  7.      }, 1000); 
  8. return () => clearInterval(id);    } 
  9.  }, [increase]); 
  10.  
  11.  // ... 

也就是說,每次編寫副作用代碼時,都要問自己它是否應該清理。計時器,頻繁請求(如上傳文件),sockets 幾乎總是需要清理。

6. 總結

從React鉤子開始的最好方法是學習如何使用它們。

但你也會遇到這樣的情況:你無法理解為什么他們的行為與你預期的不同。知道如何使用React Hook還不夠:你還應該知道何時不使用它們。

首先不要做的是有條件地渲染 Hook 或改變 Hook 調用的順序。無論Props 或狀態值是什么,React都期望組件總是以相同的順序調用Hook。

要避免的第二件事是使用過時的狀態值。要避免過時 狀態,請使用函數方式更新狀態。

不要忘記指出接受回調函數作為參數的 Hook 的依賴關系:例如useEffect(callback, deps),useCallback(callback, deps),這可以解決過時閉包問題。

不要將基礎結構數據(例如有關組件渲染周期,setTimeout()或setInterval())存儲到狀態中。經驗法則是將此類數據保存在 Ref 中。

最后,別忘了清除你的副作用。

~完,我是小智,我要去刷碗了。

作者:Shadeed 譯者:前端小智 來源:dmitripavlutin原文:https://dmitripavlutin.com/react-hooks-mistakes-to-avoid/

 本文轉載自微信公眾號「大遷世界」,可以通過以下二維碼關注。轉載本文請聯系大遷世界公眾號。

 

責任編輯:武曉燕 來源: 大遷世界
相關推薦

2017-08-17 09:07:45

Python編程代碼

2017-08-29 11:05:00

Python編程錯誤

2021-04-29 15:29:52

機器學習人工智能AI

2023-01-09 15:16:17

2021-04-22 08:00:00

人工智能機器學習數據

2017-08-02 16:47:43

數據數據收集數據分析

2021-02-24 07:40:38

React Hooks閉包

2021-12-02 18:07:53

云網絡部署云端云計算

2023-05-11 09:06:50

錯誤IT培訓

2018-03-17 09:04:35

2022-10-10 09:00:35

ReactJSX組件

2023-06-02 14:24:23

ChatGPT人工智能

2021-12-03 15:00:18

人工智能自然語言機器學習

2024-01-26 06:33:06

數據策略決策

2018-07-11 05:24:05

機器學習人工智能數據

2018-04-25 06:21:57

多云云計算IT

2022-03-08 09:31:48

云配置云安全

2013-04-23 10:57:27

iOS開發App icon設計

2021-06-28 10:12:34

云計算云平臺云計算架構

2013-08-27 14:44:05

App icon設計ASO應用商店優化app營銷推廣
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产免费一区二区 | 99精品视频免费观看 | 人人看人人草 | 午夜在线影院 | 免费在线观看一区二区三区 | 插插插干干干 | 亚洲精品永久免费 | a毛片视频网站 | 日日夜夜精品免费视频 | 久久国产精品久久 | av网站免费| 日韩精品在线一区 | 美女爽到呻吟久久久久 | 亚洲成人精品久久 | 久久久久精 | 日本福利在线观看 | 99热.com| 亚洲精品一区中文字幕乱码 | 91毛片在线观看 | 国产xxxx在线 | 国产线视频精品免费观看视频 | 久久久久国产精品午夜一区 | 久久国产精品99久久久久久丝袜 | 一区二区精品 | 五月天婷婷久久 | 秋霞电影一区二区 | 草草草影院 | 69xxx免费| 欧美成人一区二区 | 成人午夜 | 91嫩草精品 | 国产有码 | 91精品国产777在线观看 | 亚洲一区二区精品视频 | 一区二区在线 | a在线观看免费 | h片免费在线观看 | 欧美一级二级三级视频 | 欧美精品一区二区在线观看 | 色播久久| 精品欧美一区二区三区精品久久 |