一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

node.js|vue.js|jquery|angularjs|React|json|js教程|

服務器之家 - 編程語言 - JavaScript - React - 使用hooks寫React組件需要注意的5個地方

使用hooks寫React組件需要注意的5個地方

2022-02-24 16:45forrest醬 React

這篇文章主要介紹了使用hooks寫React組件需要注意的5個地方,幫助大家更好的理解和學習使用React組件,感興趣的朋友可以了解下

Hook是React16.8開始新增的特性。雖然React官方文檔已經作出了針對React hooks的相關概念的講解,但是光看官方文檔是很難將hooks使用好的,在編寫hooks的過程中很容易跳進陷阱和錯誤。本文總結了5個不好的地方。

01.不需要render的場景下使用useState

在函數組件中我們可以使用useState來管理狀態,這使得對狀態的管理變得很簡單,但是也容易被濫用,我們通過下面的代碼樣例看下容易忽略的地方。

不推薦×

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
function ClickButton(props){
 const [count, setCount] = setState(0)
 const onClickCount = () => {
  setCount((c) => c + 1)
 }
 const onClickRequest = () => {
  apiCall(count)
 }
 
 return (
  <div>
   <button onClick={onClickCount}>Click</button>
   <button onClick={onClickRequest}>Submit</button>
  </div>
 )
}

問題所在:仔細看上面的代碼,乍一看其實也沒什么問題,點擊按鈕更新 count。但是問題也就出在這里,我們的 return 部分并沒有用到 count 狀態,而每次 setCount 都會使組件重新渲染一次,而這個渲染并不是我們需要的,多出來的渲染會使得頁面的性能變差,因此我們可以改造一下代碼,如下代碼:

推薦√
如果我們只是單純的想要一個能在組件聲明周期內保存的變量,但是變量的更新不需要組件的重新渲染,我們可以使用 useRef 鉤子。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
function ClickButton(props){
 const count = useRef(0)
 const onClickCount = () => {
  count.current++
 }
 const onClickRequest = () => {
  apiCall(count.current)
 }
 
 return (
  <div>
   <button onClick={onClickCount}>Click</button>
   <button onClick={onClickRequest}>Submit</button>
  </div>
 )
}

02.使用了router.push而非link

在React SPA應用中,我們用react-router來處理路由的跳轉,我們很經常在組件中寫了一個按鈕,通過點擊按鈕的事件來處理路由的跳轉,如下代碼:

不推薦×

?
1
2
3
4
5
6
7
function ClickButton(props){
 const history = useHistory()
 const onClickGo = () => {
  history.push('/where-page')
 }
 return <button onClick={onClickGo}>Go to where</button>
}

問題所在:盡管上述代碼可以正常工作,但是卻不符合Accessibility(易訪問性設計)的要求,此類按鈕并不會被屏幕閱讀器當作一個可以跳轉的鏈接。因此我們可以改造一下代碼,如下代碼:

推薦√

?
1
2
3
4
5
function ClickButton(props){
 return <Link to="/next-page">
  <span>Go to where</span>
 </Link>
}

03.通過useEffect來處理actions

有時候,我們只想在 React 更新 DOM 之后運行一些額外的代碼。比如發送網絡請求,手動變更 DOM,記錄日志。

不推薦×

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
function DataList({ onSuccess }) {
 const [loading, setLoading] = useState(false);
 const [error, setError] = useState(null);
 const [data, setData] = useState(null);
 
 const fetchData = () => {
  setLoading(true);
  callApi()
   .then((res) => setData(res))
   .catch((err) => setError(err))
   .finally(() => setLoading(false));
 };
 
 useEffect(() => {
  fetchData();
 }, []);
 
 useEffect(() => {
  if (!loading && !error && data) {
   onSuccess();
  }
 }, [loading, error, data, onSuccess]);
 
 return <div>Data: {data}</div>;
}

問題所在:上面的代碼使用了兩個useEffect ,第一個用來請求異步數據,第二個用來調用回調函數。在第一個異步請求數據成功,才會觸發第二個 useEffect 的執行,但是,我們并不能完全保證,第二個 useEffect 的依賴項完全受控于第一個 useEffect 的成功請求數據。因此我們可以改造一下代碼,如下代碼:

推薦√

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
function DataList({ onSuccess }) {
 const [loading, setLoading] = useState(false);
 const [error, setError] = useState(null);
 const [data, setData] = useState(null);
 
 const fetchData = () => {
  setLoading(true);
  callApi()
   .then((res) => {
    setData(res)
    onSuccess()
    })
   .catch((err) => setError(err))
   .finally(() => setLoading(false));
 };
 
 useEffect(() => {
  fetchData();
 }, []);
 return <div>Data: {data}</div>;
}

04.單一職責組件

什么時候該把一個組件分成幾個更小的組件?如何構建組件樹?在使用基于組件的框架時,所有這些問題每天都會出現。然而,設計組件時的一個常見錯誤是將兩個用例組合成一個組件。

不推薦×

?
1
2
3
4
5
6
7
8
9
10
11
function Header({ menuItems }) {
 return (
  <header>
   <HeaderInner menuItems={menuItems} />
  </header>
 );
}
 
function HeaderInner({ menuItems }) {
 return isMobile() ? <BurgerButton menuItems={menuItems} /> : <Tabs tabData={menuItems} />;
}

問題所在:上面的代碼通過這種方法,組件HeaderInner試圖同時成為兩個不同的東西,一次做不止一件事情并不是很理想。此外,它還使得在其他地方測試或重用組件變得更加困難。因此我們可以改造一下代碼,如下代碼:

推薦√

將條件提升一級,可以更容易地看到組件的用途,并且它們只有一個職責,即<Tabs/><BurgerButton/>,而不是試圖同時成為兩個不同的東西。

?
1
2
3
4
5
6
7
function Header(props) {
 return (
  <header>
   {isMobile() ? <BurgerButton menuItems={menuItems} /> : <Tabs tabData={menuItems} />}
  </header>
 )
}

05.單一職責useEffects

通過對比componentWillReceivePropscomponentDidUpdate方法,才認識到userEffect的美麗。但是沒有妥當使用useEffect也是容易出問題的。

不推薦×

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
function Example(props) {
 const location = useLocation();
 const fetchData = () => {
  /* Calling the api */
 };
 
 const updateBreadcrumbs = () => {
  /* Updating the breadcrumbs*/
 };
 
 useEffect(() => {
  fetchData();
  updateBreadcrumbs();
 }, [location.pathname]);
 
 return (
  <div>
   <BreadCrumbs />
  </div>
 );
}

問題所在:上面的useEffect同時觸發了兩個副作用,但是并不都是我們需要的副作用,因此我們可以改造一下代碼,如下代碼:

推薦√

將兩個副作用從一個useEffect中分離出來。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
function Example(props) {
 const location = useLocation();
 
 const fetchData = () => {
  /* Calling the api */
 };
 
 const updateBreadcrumbs = () => {
  /* Updating the breadcrumbs*/
 };
 
 useEffect(() => {
  fetchData();
  updateBreadcrumbs();
 }, [location.pathname]);
 
 return (
  <div>
   <BreadCrumbs />
  </div>
 );
}

參考:

Five common mistakes writing react components (with hooks) in 2020

以上就是使用hooks寫React組件需要注意的5個地方的詳細內容,更多關于hooks寫React組件的資料請關注服務器之家其它相關文章!

原文鏈接:https://juejin.cn/post/6947573426076778533

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 免费观看的毛片 | 日韩大片免费观看 | 91国内精品久久久久影院优播 | 91国内精品久久久久影院优播 | 美女艹b| 日韩无砖专区2020在线 | 范冰冰上面好大下面好紧 | 天天操天天干天天舔 | 日本特级a禁片在线播放 | sihu国产午夜精品一区二区三区 | 韩国美女豪爽一级毛片 | 男人把大ji巴放进男人免费视频 | 国产成人精品第一区二区 | 久久re亚洲在线视频 | 精品高潮呻吟99AV无码 | 欧美日韩国产亚洲一区二区三区 | 国产永久一区二区三区 | 99国产精品免费观看视频 | 大片毛片女女女女女女女 | 午夜影院和视费x看 | 国模李丽莎大尺度啪啪 | 乌克兰肥熟 | 日本一区二区三区在线 观看网站 | 成人欧美一区二区三区黑人 | 亚洲国产精品久久网午夜小说 | 欧美成人免费草草影院视频 | 亚洲日本视频在线观看 | 日本成人黄色网址 | 新影音先锋男人色资源网 | 国产成人免费在线观看 | 免费观看a毛片一区二区不卡 | 四虎永久免费地址ww417 | pregnanthd产子| 波多野结衣在线中文 | 国语刺激对白勾搭视频在线观看 | 国产一区在线 | 人与善xuanwen在线400 | 久久视频精品3线视频在线观看 | 好大好爽好涨太深了小喜 | 久久99精国产一区二区三区四区 | free性泰国女人hd |