React 是一个专注的组件库。因此,它对如何请求远程数据没有什么建议。如果要通过 HTTP 请求数据并将其发送到 Web API,可以考虑下面四种方法。
为德州等地区用户提供了全套网页设计制作服务,及德州网站建设行业解决方案。主营业务为成都网站设计、成都网站建设、德州网站设计,以传统方式定制建设网站,并提供域名空间备案等一条龙服务,秉承以专业、用心的态度为用户提供真诚的服务。我们深信只要达到每一位用户的要求,就会得到认可,从而选择与我们长期合作。这样,我们也可以走得更远!
注意:在本文中,我将使用 fetch 进行 HTTP 调用,但是这些模式也适用于 Axios 之类的替代方法。另外,如果你使用的是 GraphQ L,还可以考虑使用 Apollo 之类的其他不错的选择。这篇文章假设你正在调用传统的 REST API。
方式1:内联
这是最简单,最直接的选择。在 React 组件中进行 HTTP 调用并处理响应。
- fetch("/users").then(response => response.json());
看起来很简单。但是这个示例忽略了加载状态,错误处理,声明和设置相关状态等。在现实世界中, HTTP 调用看起来更像这样。
- 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;
- }
对于一个简单的应用程序,只要发起几个请求,就可以正常工作。但是上面的状态声明和 useEffect 都是模版。如果我要进行许多 HTTP 调用,我不想为每个调用重复和维护大约 20 行代码。内联调用让你的代码变得很丑。
看一下我们要解决的一些问题:
这只是一个简单的示例,它忽略了许多其他相关问题。
方式2:文件夹集中管理
如果我们在一个文件夹中处理所有 HTTP 调用会怎么样? 使用这种方法,我们创建了一个名为 services 的文件夹,并且把进行 HTTP 调用的函数都放进去。service 是比较流行的术语,我在下面也讨论了很多好的替代名称,如 client 或 api。
要点是,所有的 HTTP 调用都是通过纯 JavaScript 函数处理的,存储在一个文件夹中。这是一个集中的 getUsers 函数:
- export function getUsers() {
- return fetch(`${process.env.REACT_APP_API_BASE_URL}users`).then(response =>
- response.json()
- );
- }
下面是对 getUsers 函数的调用:
- 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;
- }
然而这并没有太简化请求调用。主要的好处是它可以强制一致地处理 HTTP 调用。其思想是这样的:当相关函数一起处理时,更容易一致地处理它们。如果 userService 文件夹中充满了进行 HTTP 调用的函数,那么我可以很容易地确保它们始终如一地这样做。此外,如果调用被复用,则很容易从这个集中位置调用它们。
然而,我们还可以做得更好。
方式3:自定义Hook
借助 React Hooks 的魔力,我们终于可以集中处理重复的逻辑。那么如何创建一个自定义 useFetch 钩子来简化我们的 HTTP 调用呢?
- 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 };
- }
你的可能看起来不一样,但我发现这个基本的使用方法很有用。这个 Hook 极大地简化了所有调用。看看使用这个 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;
- }
对于许多应用程序,你只需要一个这样的自定义Hook。但是这个Hook已经很复杂了,并且它消除了许多问题。
但是还有很多我们没有考虑到的点:缓存?、如果客户端的连接不可靠,如何重新获取?你想在用户重新调整标签时重新获取新数据吗?如何消除重复查询?
你可以不断完善这个自定义Hook来完成所有这些操作。但是,您应该只需要方式4:
方式4:react-query/swr
使用 react-query或swr,可以为我们处理缓存、重试、重复查询等等。我不必维护自己的自定义Hook了。而且每个 HTTP 调用都需要很少的代码:
- 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;
- }
对于大多数应用程序来说,今天这是我的选择。这是完整的代码:https://codesandbox.io/s/4-ways-to-handle-restful-http-in-react-k3xug,你可以自己进行比较。
文章标题:React中请求远程数据的四种方法
网站网址:http://www.csdahua.cn/qtweb/news42/274092.html
网站建设、网络推广公司-快上网,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等
声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 快上网