<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          React 中請求遠程數(shù)據(jù)的四種方法

          共 4755字,需瀏覽 10分鐘

           ·

          2020-07-30 18:26


          React?是一個專注的組件庫。因此,它對如何請求遠程數(shù)據(jù)沒有什么建議。如果要通過?HTTP?請求數(shù)據(jù)并將其發(fā)送到?Web API,可以考慮下面四種方法。

          • 內(nèi)聯(lián)寫法
          • 集中管理
          • 自定義?Hook
          • react-query/swr

          注意:在本文中,我將使用 fetch 進行 HTTP 調(diào)用,但是這些模式也適用于 Axios 之類的替代方法。另外,如果你使用的是 GraphQ L,還可以考慮使用 Apollo 之類的其他不錯的選擇。這篇文章假設(shè)你正在調(diào)用傳統(tǒng)的 REST API。

          方式1:內(nèi)聯(lián)

          這是最簡單,最直接的選擇。在?React?組件中進行?HTTP?調(diào)用并處理響應(yīng)。

          fetch("/users").then(response?=>?response.json());

          看起來很簡單。但是這個示例忽略了加載狀態(tài),錯誤處理,聲明和設(shè)置相關(guān)狀態(tài)等。在現(xiàn)實世界中,?HTTP?調(diào)用看起來更像這樣。

          import?React,?{?useState,?useEffect?}?from?"react";

          export?default?function?InlineDemo()?{
          ??const?[users,?setUsers]?=?useState([]);
          ??const?[loading,?setLoading]?=?useState(true);
          ??const?[error,?setError]?=?useState(null);

          ??useEffect(()?=>?{
          ????fetch(`${process.env.REACT_APP_API_BASE_URL}users`)
          ??????.then(response?=>?{
          ????????if?(response.ok)?return?response.json();
          ????????throw?response;
          ??????})
          ??????.then(json?=>?{
          ????????setUsers(json);
          ??????})
          ??????.catch(err?=>?{
          ????????console.error(err);
          ????????setError(err);
          ??????})
          ??????.finally(()?=>?{
          ????????setLoading(false);
          ??????});
          ??},?[]);

          ??if?(loading)?return?"Loading...";
          ??if?(error)?return?"Oops!";
          ??return?users[0].username;
          }

          對于一個簡單的應(yīng)用程序,只要發(fā)起幾個請求,就可以正常工作。但是上面的狀態(tài)聲明和?useEffect?都是模版。如果我要進行許多?HTTP?調(diào)用,我不想為每個調(diào)用重復(fù)和維護大約 20 行代碼。內(nèi)聯(lián)調(diào)用讓你的代碼變得很丑。

          看一下我們要解決的一些問題:

          • 聲明加載狀態(tài)
          • 聲明錯誤狀態(tài)
          • 將錯誤打印到控制臺
          • 檢查響應(yīng)是否通過返回 200?response.ok
          • 如果響應(yīng)正常,將響應(yīng)轉(zhuǎn)換為?json?并返回?promise
          • 如果響應(yīng)不正確,拋出錯誤
          • 在?finally?中隱藏加載狀態(tài),以確保?Loading?即使發(fā)生錯誤也被隱藏
          • 聲明一個空的依賴項數(shù)組,以便?useEffect?只運行一次

          這只是一個簡單的示例,它忽略了許多其他相關(guān)問題。

          方式2:文件夾集中管理

          如果我們在一個文件夾中處理所有?HTTP?調(diào)用會怎么樣? 使用這種方法,我們創(chuàng)建了一個名為?services?的文件夾,并且把進行 HTTP 調(diào)用的函數(shù)都放進去。service?是最流行的術(shù)語,我在下面也討論了很多好的替代名稱,如?client?或?api

          要點是,所有的?HTTP?調(diào)用都是通過純?JavaScript?函數(shù)處理的,存儲在一個文件夾中。這是一個集中的?getUsers?函數(shù):

          export?function?getUsers()?{
          ??return?fetch(`${process.env.REACT_APP_API_BASE_URL}users`).then(response?=>
          ????response.json()
          ??);
          }

          下面是對?getUsers?函數(shù)的調(diào)用:

          import?React,?{?useState,?useEffect?}?from?"react";
          import?{?getUsers?}?from?"./services/userService";

          export?default?function?CentralDemo()?{
          ??const?[users,?setUsers]?=?useState([]);
          ??const?[loading,?setLoading]?=?useState(true);
          ??const?[error,?setError]?=?useState(null);
          ??useEffect(()?=>?{
          ????getUsers()
          ??????.then(json?=>?{
          ????????setUsers(json);
          ????????setLoading(false);
          ??????})
          ??????.catch(err?=>?{
          ????????console.error(err);
          ????????setError(err);
          ??????});
          ??},?[]);

          ??if?(loading)?return?"Loading...";
          ??if?(error)?return?"Oops!";
          ??return?users[0].username;
          }

          然而這并沒有太簡化請求調(diào)用。主要的好處是它可以強制一致地處理?HTTP?調(diào)用。其思想是這樣的:當(dāng)相關(guān)函數(shù)一起處理時,更容易一致地處理它們。如果?userService?文件夾中充滿了進行?HTTP?調(diào)用的函數(shù),那么我可以很容易地確保它們始終如一地這樣做。此外,如果調(diào)用被復(fù)用,則很容易從這個集中位置調(diào)用它們。

          然而,我們還可以做得更好。

          方式3:自定義Hook

          借助?React Hooks?的魔力,我們終于可以集中處理重復(fù)的邏輯。那么如何創(chuàng)建一個自定義?useFetch?鉤子來簡化我們的?HTTP?調(diào)用呢?

          import?{?useState,?useEffect,?useRef?}?from?"react";
          //?This?custom?hook?centralizes?and?streamlines?handling?of?HTTP?calls
          export?default?function?useFetch(url,?init)?{
          ??const?[data,?setData]?=?useState(null);
          ??const?[loading,?setLoading]?=?useState(true);
          ??const?[error,?setError]?=?useState(null);
          ??const?prevInit?=?useRef();
          ??const?prevUrl?=?useRef();

          ??useEffect(()?=>?{
          ??//?Only?refetch?if?url?or?init?params?change.
          ????if?(prevUrl.current?===?url?&&?prevInit.current?===?init)?return;
          ????prevUrl.current?=?url;
          ????prevInit.current?=?init;
          ????fetch(process.env.REACT_APP_API_BASE_URL?+?url,?init)
          ??????.then(response?=>?{
          ????????if?(response.ok)?return?response.json();
          ????????setError(response);
          ??????})
          ??????.then(data?=>?setData(data))
          ??????.catch(err?=>?{
          ????????console.error(err);
          ????????setError(err);
          ??????})
          ??????.finally(()?=>?setLoading(false));
          ??},?[init,?url]);

          ??return?{?data,?loading,?error?};
          }

          你的可能看起來不一樣,但我發(fā)現(xiàn)這個基本的使用方法很有用。這個 Hook 極大地簡化了所有調(diào)用。看看使用這個?Hook?需要多少代碼 :

          import?React?from?"react";
          import?useFetch?from?"./useFetch";

          export?default?function?HookDemo()?{
          ??const?{?data,?loading,?error?}?=?useFetch("users");
          ??if?(loading)?return?"Loading...";
          ??if?(error)?return?"Oops!";
          ??return?data[0].username;
          }

          對于許多應(yīng)用程序,你只需要一個這樣的自定義Hook。但是這個Hook已經(jīng)很復(fù)雜了,并且它消除了許多問題。

          但是還有很多我們沒有考慮到的點:緩存?、如果客戶端的連接不可靠,如何重新獲取?你想在用戶重新調(diào)整標(biāo)簽時重新獲取新數(shù)據(jù)嗎?如何消除重復(fù)查詢?

          你可以不斷完善這個自定義Hook來完成所有這些操作。但是,您應(yīng)該只需要方式4:

          方式4:react-query/swr

          使用?react-query或swr,可以為我們處理緩存、重試、重復(fù)查詢等等。我不必維護自己的自定義Hook了。而且每個?HTTP?調(diào)用都需要很少的代碼:

          import?React?from?"react";
          import?{?getUsers?}?from?"./services/userService";
          import?{?useQuery?}?from?"react-query";

          export?default?function?ReactQueryDemo()?{
          ??const?{?data,?isLoading,?error?}?=?useQuery("users",?getUsers);
          ??if?(isLoading)?return?"Loading...";
          ??if?(error)?return?"Oops!";
          ??return?data[0].username;
          }

          對于大多數(shù)應(yīng)用程序來說,今天這是我的首選。這是完整的代碼:https://codesandbox.io/s/4-ways-to-handle-restful-http-in-react-k3xug,你可以自己進行比較。


          - EOF -


          ?? 看完三件事

          如果你覺得這篇內(nèi)容對你挺有啟發(fā),我想邀請你幫我三個小忙:


          1. 點個「在看」,讓更多的人也能看到這篇內(nèi)容(喜歡不點在看,都是耍流氓 -_-)

          2. 關(guān)注我的官網(wǎng)?https://muyiy.cn,讓我們成為長期關(guān)系

          3. 關(guān)注公眾號「高級前端進階」,公眾號后臺回復(fù)「面試題」 送你高級前端面試題,回復(fù)「加群」加入面試互助交流群


          》》面試官都在用的題庫,快來看看《《

          瀏覽 37
          點贊
          評論
          收藏
          分享

          手機掃一掃分享

          分享
          舉報
          評論
          圖片
          表情
          推薦
          點贊
          評論
          收藏
          分享

          手機掃一掃分享

          分享
          舉報
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  91福利视频网站 | 日韩熟女一区二区三区 | 特级西西444www大精品视频免费看 | 中文字幕成人av 中文字幕第45页 中文字幕第十二页 | 伊人伊人伊人伊人伊人 |