queries(查询)

查询基础 (Query Basics)

查询(Query)是对绑定到 唯一键(unique key) 的异步数据源的声明性依赖。查询可以与任何基于 Promise 的方法(包括 GET 和 POST 方法)一起使用,以便从服务器获取数据。如果你的方法修改了服务器上的数据,我们建议改用 变更(Mutations)

要在组件或自定义 Hook 中订阅查询,请调用 useQuery hook,其中至少包含:

  • 一个 查询的唯一键

  • 一个返回 Promise 的函数,该函数:

    • 解析(Resolve)数据,或者
    • 抛出(Throw)错误
TypeScript 复制代码
import { useQuery } from '@tanstack/react-query'

function App() {
  const info = useQuery({ queryKey: ['todos'], queryFn: fetchTodoList })
}

你提供的 唯一键 在内部用于在整个应用程序中重新获取、缓存和共享你的查询。

useQuery 返回的查询结果包含了你在模板化或以其他方式使用数据时所需的所有信息:

TypeScript 复制代码
const result = useQuery({ queryKey: ['todos'], queryFn: fetchTodoList })

result 对象包含几个非常重要的状态,你需要了解它们才能高效工作。在任何给定时刻,查询只能处于以下状态之一:

  • isPendingstatus === 'pending' - 查询尚无数据
  • isErrorstatus === 'error' - 查询遇到错误
  • isSuccessstatus === 'success' - 查询成功且数据可用

除了这些主要状态之外,根据查询的状态,还有更多信息可用:

  • error - 如果查询处于 isError 状态,可以通过 error 属性获取错误信息。
  • data - 如果查询处于 isSuccess 状态,可以通过 data 属性获取数据。
  • isFetching - 在任何状态下,如果查询在任何时候正在获取数据(包括后台重新获取),isFetching 将为 true

对于 大多数 查询,通常只需检查 isPending 状态,然后是 isError 状态,最后假设数据可用并渲染成功状态即可:

TypeScript 复制代码
function Todos() {
  const { isPending, isError, data, error } = useQuery({
    queryKey: ['todos'],
    queryFn: fetchTodoList,
  })

  if (isPending) {
    return <span>加载中...</span>
  }

  if (isError) {
    return <span>错误: {error.message}</span>
  }

  // 到这里我们可以假设 `isSuccess === true`
  return (
    <ul>
      {data.map((todo) => (
        <li key={todo.id}>{todo.title}</li>
      ))}
    </ul>
  )
}

如果你不喜欢使用布尔值,你也可以随时使用 status 状态:

TypeScript 复制代码
function Todos() {
  const { status, data, error } = useQuery({
    queryKey: ['todos'],
    queryFn: fetchTodoList,
  })

  if (status === 'pending') {
    return <span>加载中...</span>
  }

  if (status === 'error') {
    return <span>错误: {error.message}</span>
  }

  // 虽然 status === 'success',但 "else" 逻辑也行得通
  return (
    <ul>
      {data.map((todo) => (
        <li key={todo.id}>{todo.title}</li>
      ))}
    </ul>
  )
}

如果你在访问 data 之前检查了 pendingerror,TypeScript 也会正确地缩小 data 的类型范围。

FetchStatus (获取状态)

除了 status 字段外,你还将获得一个额外的 fetchStatus 属性,其选项如下:

  • fetchStatus === 'fetching' - 查询当前正在获取数据。
  • fetchStatus === 'paused' - 查询想要获取数据,但被暂停了。更多信息请阅读 网络模式 指南。
  • fetchStatus === 'idle' - 查询当前没有进行任何操作。

为什么有两种不同的状态?

后台重新获取(Background refetches)和"过期时重新验证(stale-while-revalidate)"逻辑使得 statusfetchStatus 的所有组合成为可能。例如:

  • 一个处于 success 状态的查询通常 fetchStatusidle,但如果正在进行后台重新获取,它也可能处于 fetching 状态。
  • 一个刚挂载且没有数据的查询通常处于 pending 状态和 fetchingfetchStatus,但如果没有网络连接,它也可能是 paused

所以请记住,查询可能处于 pending 状态,但实际上并没有在获取数据。作为一个经验法则:

  • status 提供关于 data 的信息:我们要么有数据,要么没有。
  • fetchStatus 提供关于 queryFn 的信息:它是否正在运行?

延伸阅读

关于执行状态检查的另一种方法,请查看 TkDodo 的这篇文章

相关推荐
Justin3go2 小时前
HUNT0 上线了——尽早发布,尽早发现
前端·后端·程序员
怕浪猫3 小时前
第一章 JSX 增强特性与函数组件入门
前端·javascript·react.js
铅笔侠_小龙虾3 小时前
Emmet 常用用法指南
前端·vue
钦拆大仁3 小时前
跨站脚本攻击XSS
前端·xss
VX:Fegn08954 小时前
计算机毕业设计|基于springboot + vue校园社团管理系统(源码+数据库+文档)
前端·数据库·vue.js·spring boot·后端·课程设计
ChangYan.5 小时前
直接下载源码但是执行npm run compile后报错
前端·npm·node.js
skywalk81635 小时前
在 FreeBSD 上可以使用的虚拟主机(Web‑Hosting)面板
前端·主机·webmin
ohyeah6 小时前
深入理解 React 中的 useRef:不只是获取 DOM 元素
前端·react.js
MoXinXueWEB6 小时前
前端页面获取不到url上参数值
前端
低保和光头哪个先来6 小时前
场景6:对浏览器内核的理解
开发语言·前端·javascript·vue.js·前端框架