React Query 实战教程:在 React 中如何优雅的管理接口数据状态?

news/2024/7/15 19:01:52 标签: react.js, 前端, 前端框架

前言

如何通过Ajax或者Fetch优雅的请求后端接口,这是所有复杂前端项目都需要考虑处理的事情。在React项目中,有不少成熟的Hook能够让开发者管理整个请求过程中的数据和状态,例如 axios-hooksuse-httpreact-queryswr甚至 ahook中提供的 useRequest

我曾经很长一段时间是直接使用 ahook中的 useRequest ,但是有的项目中不需要 ahook 中的其他hook,我又不愿意仅仅为了使用 useRequest 而在项目中引入 ahook 。就像我第参与的一个前端项目中仅仅为了使用 jquery$.ajax而引入 jquery,至今我都在怀疑当时前端负责人的真实水平。

最开始听说 react-query 时,我还以为它是处理URL字符串的工具库,随着慢慢熟练使用 react-query 后发现是真的很好用。本文将详细介绍 react-query 的功能、优势以及如何使用。

React Query 介绍

React Query 是一个与数据无关的、强大的数据同步库。它通过提供一些 Hooks,可以帮助我们以简单的方式同步后端数据到我们的 React 组件中。

React Query 对于数据获取、同步、缓存、更新等操作提供了极好的解决方案。

解决问题

  1. 网络请求的缓存和更新

    React Query 自动处理缓存、数据的更新和废弃,不需要手动操作,简化操作流程。

  2. 后台更新

    React Query 可以在后台静默更新数据,不影响用户的使用体验。

  3. 调试工具

    React Query Devtools 提供了全面的调试支持,使数据流的控制和错误的追踪更为容易。

使用步骤

1. 安装依赖

npm install react-query

2. 配置全局Provider

为了让 React Query 在你的整个应用中都能工作,需要在应用的根组件设置一个**QueryClientProvider** ,React Query 中的所有 hook 运行时需要这个全局的上下文状态。

import { QueryClient, QueryClientProvider } from 'react-query'
import Todos from './Todos'

const queryClient = new QueryClient()

function App() {
  return (
    <QueryClientProvider client={queryClient}>
      <Todos />
    </QueryClientProvider>
  )
}

export default App

3. 简单的 GET 请求

使用React Query 的主要获取数据的 Hook useQueryuseQuery 是最基础的数据获取 Hook。 可以获取、缓存、同步、自动更新数据。

下面的示例中,它会请求并显示所有的 todos

import React from 'react'
import { QueryClient, QueryClientProvider, useQuery } from 'react-query'
import axios from 'axios'

const queryClient = new QueryClient()

function Todos() {
  const fetchTodos = async () => {
    const { data } = await axios.get('/api/todos')
    return data
  }

  **const { isLoading, error, data } = useQuery('todos', fetchTodos)**

  if (isLoading) return 'Loading...'

  if (error) return 'An error has occurred: ' + error.message

  return (
    <ul>
      {data.map(todo => (
        <li key={todo.id}>{todo.title}</li>
      ))}
    </ul>
  )
}

function App() {
  return (
    <QueryClientProvider client={queryClient}>
      <Todos />
    </QueryClientProvider>
  )
}

export default App

useQuery 参数详细说明

useQuery(queryKey, queryFn, options)
参数详细参数可选类型作用说明
queryKey**StringArray**
queryFnFunction这个函数会返回一个 Promise,用于获取数据。如果你的 query key 是一个数组,每个数组的元素都会作为单独的参数传递给这个函数。
optionscacheTimeNumber不活动的数据在被自动垃圾回收清除之前的毫秒数。默认是 5 分钟。
staleTimeNumber数据在获取后变得过时之前的毫秒数。默认情况下,数据在获取后就会立即变过时。
retryNumber如果查询失败,将尝试重新查询的次数。默认是 3 次。
retryDelayNumber重试查询之间的毫秒数。默认为 1000 毫秒。

进阶用法

1. 如何实现分页查询

React Query提供了**usePaginatedQuery**方便地用于实现分页查询。

import React from 'react'
import { usePaginatedQuery } from 'react-query'
import axios from 'axios'

const fetchProjects = async (key, page = 0) => {
  const res = await axios.get(`/api/projects?page=${page}`)
  return res.data
}

function Projects() {
  const [page, setPage] = React.useState(0)
  const {
    isLoading,
    isError,
    error,
    resolvedData,
    latestData,
    isFetching,
  } = usePaginatedQuery(['projects', page], fetchProjects)

  if (isLoading) return 'Loading...'
  if (isError) return `An error has occurred: ${error.message}`

  return (
    <div>
      {resolvedData.projects.map(project => (
        <p key={project.id}>{project.name}</p>
      ))}

      <div>
        <button
          onClick={() => setPage(old => Math.max(old - 1, 0))}
          disabled={page === 0}
        >
          Previous Page
        </button>{' '}
        <button
          onClick={() => { 
            if (!latestData || !latestData.hasMore) {
              return
            } 
            setPage(old => old + 1)
          }}
          disabled={!latestData || !latestData.hasMore}
        >
          Next Page
        </button>
      </div>
      {isFetching ? ' Loading more...' : null}
    </div>
  )
}

export default Projects

需要注意的点

**resolvedData是上次成功获取的数据,而 latestData则是最新尝试过获取的数据,所以当用户点击“下一页”按钮时,新的数据正在加载,但在新的数据被成功加载之前,resolvedData**仍然是上次成功获取的数据。

总结

如果你的项目技术栈使用的是React、并且你的项目需要与后端进行接口交互,那么 React Query 是很好的提效工具。React Query 提供了十多个 Hook 来应对各种复杂的业务场景,分页查询、无限加载、并发请求、接口预加载等等场景都能够完美处理。

更多的业务场景实战,我会在后续的文章中详细介绍,感兴趣的朋友可以关注一下。


http://www.niftyadmin.cn/n/5292552.html

相关文章

面试记录(被问倒的)

快排&#xff0c;设计模式&#xff08;观察者模式 mvc模式 单例模式&#xff09;及其优势&#xff0c;工作遇到的困难&#xff0c;解决过的问题&#xff0c;渲染管线&#xff0c;ui合批&#xff08;动静分离&#xff09;&#xff0c;比较复杂的编辑器扩展&#xff0c;算法&…

springcloud中使用openfeign来优化接口调用

简单介绍在springcloud中使用openfeign来优化接口调用 目录 一、引入依赖二、为服务提供者编写openfeign接口三、服务消费者调用定义的openfeign接口四、项目结构五、日志级别配置1、通过配置类进行全局配置2、通过配置类进行局部配置3、通过配置文件配置 一、引入依赖 <!-…

C++高级-模板详解

目录 一、概述 二、函数模板 2.1格式 2.2函数模板的实例化

kafka容灾演练的方案

背景 kafka可以通过MirrorMaker工具把集群的数据从一个集群同步到另一个集群&#xff0c;通过在另一个数据中心创建灾备集群的方式可以做到容灾的效果,但是如果我们不通过如此重量级的工具也想达到容灾演练的目的&#xff0c;可以怎么做呢 kafka简单容灾实现 当原kafka集群发…

跨境电商ERP源码的10大利弊及解决方案

全球跨境电商行业的快速发展&#xff0c;促使电商企业迫切需要一套高效的管理工具来应对多元化的业务需求。跨境电商ERP源码因其强大的功能和方便的管理方式逐渐成为电商企业的首选工具之一。然而&#xff0c;尽管跨境电商ERP源码带来了许多利益&#xff0c;却也伴随着一些潜在…

深入浅出理解转置卷积Conv2DTranspose

温故而知新&#xff0c;可以为师矣&#xff01; 一、参考资料 论文&#xff1a;A guide to convolution arithmetic for deep learning github源码&#xff1a;Convolution arithmetic bilibili视频&#xff1a;转置卷积&#xff08;transposed convolution&#xff09; 转置…

数据集介绍【02】CIFAR10

CIFAR10数据集共有60000个样本&#xff0c;每个样本都是一张32*32像素的RGB图像&#xff08;彩色图像&#xff09;&#xff0c;每个RGB图像又必定分为3个通道&#xff08;R通道、G通道、B通道&#xff09;。这60000个样本被分成了50000个训练样本和10000个测试样本。 CIFAR10数…

Excel formulas 使用总结(更新中)

最近在写task assigment的时候学习到的&#xff0c;记录下。 首先它所有需要写赋值formuls都要用 开头 相等赋值 a1 这个就代表这格的数据和a1是一样的。如果希望其他格和它相同的逻辑&#xff0c;可以直接复制该cell或者直接拖动该cell右下角&#xff0c;他会自动进行匹配…