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

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

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

 [[334859]]

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 之類的其他不錯的選擇。這篇文章假設你正在調(diào)用傳統(tǒng)的 REST API。

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

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

 

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

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

 

  1. import React, { useState, useEffect } from "react"
  2.  
  3. export default function InlineDemo() { 
  4.   const [users, setUsers] = useState([]); 
  5.   const [loading, setLoading] = useState(true); 
  6.   const [error, setError] = useState(null); 
  7.  
  8.   useEffect(() => { 
  9.     fetch(`${process.env.REACT_APP_API_BASE_URL}users`) 
  10.       .then(response => { 
  11.         if (response.ok) return response.json(); 
  12.         throw response; 
  13.       }) 
  14.       .then(json => { 
  15.         setUsers(json); 
  16.       }) 
  17.       .catch(err => { 
  18.         console.error(err); 
  19.         setError(err); 
  20.       }) 
  21.       .finally(() => { 
  22.         setLoading(false); 
  23.       }); 
  24.   }, []); 
  25.  
  26.   if (loading) return "Loading..."
  27.   if (error) return "Oops!"
  28.   return users[0].username; 

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

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

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

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

方式2:文件夾集中管理

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

 

 

 

 

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

 

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

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

 

  1. import React, { useState, useEffect } from "react"
  2. import { getUsers } from "./services/userService"
  3.  
  4. export default function CentralDemo() { 
  5.   const [users, setUsers] = useState([]); 
  6.   const [loading, setLoading] = useState(true); 
  7.   const [error, setError] = useState(null); 
  8.   useEffect(() => { 
  9.     getUsers() 
  10.       .then(json => { 
  11.         setUsers(json); 
  12.         setLoading(false); 
  13.       }) 
  14.       .catch(err => { 
  15.         console.error(err); 
  16.         setError(err); 
  17.       }); 
  18.   }, []); 
  19.  
  20.   if (loading) return "Loading..."
  21.   if (error) return "Oops!"
  22.   return users[0].username; 

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

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

方式3:自定義Hook

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

 

  1. import { useState, useEffect, useRef } from "react"
  2. // This custom hook centralizes and streamlines handling of HTTP calls 
  3. export default function useFetch(url, init) { 
  4.   const [data, setData] = useState(null); 
  5.   const [loading, setLoading] = useState(true); 
  6.   const [error, setError] = useState(null); 
  7.   const prevInit = useRef(); 
  8.   const prevUrl = useRef(); 
  9.  
  10.   useEffect(() => { 
  11.   // Only refetch if url or init params change. 
  12.     if (prevUrl.current === url && prevInit.current === init) return
  13.     prevUrl.current = url; 
  14.     prevInit.current = init; 
  15.     fetch(process.env.REACT_APP_API_BASE_URL + url, init) 
  16.       .then(response => { 
  17.         if (response.ok) return response.json(); 
  18.         setError(response); 
  19.       }) 
  20.       .then(data => setData(data)) 
  21.       .catch(err => { 
  22.         console.error(err); 
  23.         setError(err); 
  24.       }) 
  25.       .finally(() => setLoading(false)); 
  26.   }, [init, url]); 
  27.  
  28.   return { data, loading, error }; 

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

 

  1. import React from "react"
  2. import useFetch from "./useFetch"
  3.  
  4. export default function HookDemo() { 
  5.   const { data, loading, error } = useFetch("users"); 
  6.   if (loading) return "Loading..."
  7.   if (error) return "Oops!"
  8.   return data[0].username; 

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

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

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

方式4:react-query/swr

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

 

  1. import React from "react"
  2. import { getUsers } from "./services/userService"
  3. import { useQuery } from "react-query"
  4.  
  5. export default function ReactQueryDemo() { 
  6.   const { data, isLoading, error } = useQuery("users", getUsers); 
  7.   if (isLoading) return "Loading..."
  8.   if (error) return "Oops!"
  9.   return data[0].username; 

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

責任編輯:華軒 來源: code秘密花園
相關推薦

2022-11-04 13:35:29

IT遠程工作混合工作

2024-10-16 18:09:54

2021-03-10 10:13:39

爬蟲Python代碼

2014-03-17 09:22:43

Linux命令

2022-09-02 14:29:01

JavaScrip數(shù)組屬性

2011-07-06 18:07:16

ASP

2010-08-31 15:51:51

DB2清除數(shù)據(jù)

2023-03-17 10:51:26

2009-11-23 15:57:51

PHP偽靜態(tài)

2022-07-15 14:43:21

數(shù)據(jù)安全Linux

2011-06-22 15:21:08

XML

2020-08-10 00:30:55

備份密碼iPhone移動安全

2009-02-25 09:52:14

類型轉換.NET 強制轉型

2009-03-31 13:12:30

解析XMLJava

2011-08-02 17:26:38

Oracle數(shù)據(jù)庫遠程連接

2018-03-05 22:25:21

數(shù)據(jù)中心降低成本停機

2022-12-07 10:28:22

2016-06-28 10:19:31

云計算云安全

2010-07-16 13:50:53

Perl哈希表

2009-09-17 16:55:58

C#組件設計
點贊
收藏

51CTO技術棧公眾號