React 中请求远程数据的四种方法

开发 前端
React 是一个专注的组件库。因此,它对如何请求远程数据没有什么建议。如果要通过 HTTP 请求数据并将其发送到 Web API,可以考虑下面四种方法。

 [[334859]]

React 是一个专注的组件库。因此,它对如何请求远程数据没有什么建议。如果要通过 HTTP 请求数据并将其发送到 Web API,可以考虑下面四种方法。

  • 内联写法
  • 集中管理
  • 自定义 Hook
  • react-query/swr

注意:在本文中,我将使用 fetch 进行 HTTP 调用,但是这些模式也适用于 Axios 之类的替代方法。另外,如果你使用的是 GraphQ L,还可以考虑使用 Apollo 之类的其他不错的选择。这篇文章假设你正在调用传统的 REST API。

方式1:内联

这是最简单,最直接的选择。在 React 组件中进行 HTTP 调用并处理响应。

 

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

看起来很简单。但是这个示例忽略了加载状态,错误处理,声明和设置相关状态等。在现实世界中, HTTP 调用看起来更像这样。

 

  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; 

对于一个简单的应用程序,只要发起几个请求,就可以正常工作。但是上面的状态声明和 useEffect 都是模版。如果我要进行许多 HTTP 调用,我不想为每个调用重复和维护大约 20 行代码。内联调用让你的代码变得很丑。

看一下我们要解决的一些问题:

  • 声明加载状态
  • 声明错误状态
  • 将错误打印到控制台
  • 检查响应是否通过返回 200 response.ok
  • 如果响应正常,将响应转换为 json 并返回 promise
  • 如果响应不正确,抛出错误
  • 在 finally 中隐藏加载状态,以确保 Loading 即使发生错误也被隐藏
  • 声明一个空的依赖项数组,以便 useEffect 只运行一次

这只是一个简单的示例,它忽略了许多其他相关问题。

方式2:文件夹集中管理

如果我们在一个文件夹中处理所有 HTTP 调用会怎么样? 使用这种方法,我们创建了一个名为 services 的文件夹,并且把进行 HTTP 调用的函数都放进去。service 是比较流行的术语,我在下面也讨论了很多好的替代名称,如 client 或 api。

 

 

 

 

要点是,所有的 HTTP 调用都是通过纯 JavaScript 函数处理的,存储在一个文件夹中。这是一个集中的 getUsers 函数:

 

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

下面是对 getUsers 函数的调用:

 

  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; 

然而这并没有太简化请求调用。主要的好处是它可以强制一致地处理 HTTP 调用。其思想是这样的:当相关函数一起处理时,更容易一致地处理它们。如果 userService 文件夹中充满了进行 HTTP 调用的函数,那么我可以很容易地确保它们始终如一地这样做。此外,如果调用被复用,则很容易从这个集中位置调用它们。

然而,我们还可以做得更好。

方式3:自定义Hook

借助 React Hooks 的魔力,我们终于可以集中处理重复的逻辑。那么如何创建一个自定义 useFetch 钩子来简化我们的 HTTP 调用呢?

 

  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 }; 

你的可能看起来不一样,但我发现这个基本的使用方法很有用。这个 Hook 极大地简化了所有调用。看看使用这个 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已经很复杂了,并且它消除了许多问题。

但是还有很多我们没有考虑到的点:缓存?、如果客户端的连接不可靠,如何重新获取?你想在用户重新调整标签时重新获取新数据吗?如何消除重复查询?

你可以不断完善这个自定义Hook来完成所有这些操作。但是,您应该只需要方式4:

方式4:react-query/swr

使用 react-query或swr,可以为我们处理缓存、重试、重复查询等等。我不必维护自己的自定义Hook了。而且每个 HTTP 调用都需要很少的代码:

 

  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; 

对于大多数应用程序来说,今天这是我的首选。这是完整的代码: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数组属性

2011-07-06 18:07:16

ASP

2023-03-17 10:51:26

2010-08-31 15:51:51

DB2清除数据

2022-07-15 14:43:21

数据安全Linux

2009-11-23 15:57:51

PHP伪静态

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数据库远程连接

2018-03-05 22:25:21

数据中心降低成本停机

2022-12-07 10:28:22

2020-01-21 19:15:23

漏洞安全IT

2021-09-03 11:24:04

云计算云计算环境云应用

2010-08-02 16:47:46

Flex
点赞
收藏

51CTO技术栈公众号