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 的这篇文章

相关推荐
创码小奇客2 小时前
Trae Solo模式实战:我用3小时撸了个儿童睡前故事网页
前端·javascript·人工智能
Jing_Rainbow2 小时前
【AI-9/Lesson30(2025-11-12)】AI + Vibe Coding:Hulk 浏览器扩展开发全解析 —— 从需求文档到实战的完整指南🌈
前端·人工智能·程序员
wordbaby2 小时前
important-defaults(重要的默认配置)
前端·react.js
niucloud-admin2 小时前
diy自定义组件/页面装修开发——自定义页面模板
前端
wordbaby2 小时前
React Native 实战:构建一个现代化的 Todo List (React Query + SafeArea + 键盘适配)
前端·react native
T___T2 小时前
class 出现前,JS 是怎么继承的
前端·javascript·面试
b***74882 小时前
前端技术的下一场革命:体验、架构与智能协作的深度重构
前端·重构·架构
2201_757830872 小时前
JS的学习
前端·javascript·学习
鸡腿大王2 小时前
震惊:v8引擎竟是如此操作代码(JS预编译)
前端·javascript