查询基础 (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 对象包含几个非常重要的状态,你需要了解它们才能高效工作。在任何给定时刻,查询只能处于以下状态之一:
isPending或status === 'pending'- 查询尚无数据isError或status === 'error'- 查询遇到错误isSuccess或status === '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 之前检查了 pending 和 error,TypeScript 也会正确地缩小 data 的类型范围。
FetchStatus (获取状态)
除了 status 字段外,你还将获得一个额外的 fetchStatus 属性,其选项如下:
fetchStatus === 'fetching'- 查询当前正在获取数据。fetchStatus === 'paused'- 查询想要获取数据,但被暂停了。更多信息请阅读 网络模式 指南。fetchStatus === 'idle'- 查询当前没有进行任何操作。
为什么有两种不同的状态?
后台重新获取(Background refetches)和"过期时重新验证(stale-while-revalidate)"逻辑使得 status 和 fetchStatus 的所有组合成为可能。例如:
- 一个处于
success状态的查询通常fetchStatus为idle,但如果正在进行后台重新获取,它也可能处于fetching状态。 - 一个刚挂载且没有数据的查询通常处于
pending状态和fetching的fetchStatus,但如果没有网络连接,它也可能是paused。
所以请记住,查询可能处于 pending 状态,但实际上并没有在获取数据。作为一个经验法则:
status提供关于data的信息:我们要么有数据,要么没有。fetchStatus提供关于queryFn的信息:它是否正在运行?
延伸阅读
关于执行状态检查的另一种方法,请查看 TkDodo 的这篇文章。