如果您希望禁用查询自动运行,可以使用 enabled = false 选项。 enabled 选项也接受返回布尔值的回调函数。
当 enabled 为 false 时
TypeScript 用户可能更喜欢使用 skipToken 作为 enabled = false 的替代方案。
function Todos() {
const { isLoading, isError, data, error, refetch, isFetching } = useQuery({
queryKey: ['todos'],
queryFn: fetchTodoList,
enabled: false,
})
return (
<div>
<button onClick={() => refetch()}>Fetch Todos</button>
{data ? (
<>
<ul>
{data.map((todo) => (
<li key={todo.id}>{todo.title}</li>
))}
</ul>
</>
) : isError ? (
<span>Error: {error.message}</span>
) : isLoading ? (
<span>Loading...</span>
) : (
<span>Not ready ...</span>
)}
<div>{isFetching ? 'Fetching...' : null}</div>
</div>
)
}
function Todos() {
const { isLoading, isError, data, error, refetch, isFetching } = useQuery({
queryKey: ['todos'],
queryFn: fetchTodoList,
enabled: false,
})
return (
<div>
<button onClick={() => refetch()}>Fetch Todos</button>
{data ? (
<>
<ul>
{data.map((todo) => (
<li key={todo.id}>{todo.title}</li>
))}
</ul>
</>
) : isError ? (
<span>Error: {error.message}</span>
) : isLoading ? (
<span>Loading...</span>
) : (
<span>Not ready ...</span>
)}
<div>{isFetching ? 'Fetching...' : null}</div>
</div>
)
}
永久禁用查询会放弃 TanStack Query 提供的许多强大功能(例如后台重新获取),而且也不是惯用的方法。它将您从声明式方法(定义查询应何时运行的依赖项)带入命令式模式(每当我点击此处时获取)。也不可能将参数传递给 refetch。通常,您想要的只是延迟初始获取的懒加载查询
enabled 选项不仅可以用于永久禁用查询,还可以用于稍后启用/禁用它。一个很好的例子是过滤器表单,您只想在用户输入过滤器值后才触发第一个请求
function Todos() {
const [filter, setFilter] = React.useState('')
const { data } = useQuery({
queryKey: ['todos', filter],
queryFn: () => fetchTodos(filter),
// ⬇️ disabled as long as the filter is empty
enabled: !!filter,
})
return (
<div>
// 🚀 applying the filter will enable and execute the query
<FiltersForm onApply={setFilter} />
{data && <TodosTable data={data} />}
</div>
)
}
function Todos() {
const [filter, setFilter] = React.useState('')
const { data } = useQuery({
queryKey: ['todos', filter],
queryFn: () => fetchTodos(filter),
// ⬇️ disabled as long as the filter is empty
enabled: !!filter,
})
return (
<div>
// 🚀 applying the filter will enable and execute the query
<FiltersForm onApply={setFilter} />
{data && <TodosTable data={data} />}
</div>
)
}
懒加载查询从一开始就处于 status: 'pending' 状态,因为 pending 意味着还没有数据。从技术上讲这是正确的,但是,由于我们目前没有获取任何数据(因为查询未启用),这也意味着您可能无法使用此标志来显示加载指示器。
如果您正在使用禁用或懒加载查询,则可以使用 isLoading 标志来代替。它是一个派生标志,由以下项计算得出
isPending && isFetching
因此,只有当查询当前是首次获取时,它才为真。
如果您正在使用 TypeScript,则可以使用 skipToken 来禁用查询。当您想要根据条件禁用查询,但仍希望保持查询的类型安全时,这非常有用。
重要提示:来自 useQuery 的 refetch 不适用于 skipToken。除此之外,skipToken 的工作方式与 enabled: false 相同。
import { skipToken, useQuery } from '@tanstack/react-query'
function Todos() {
const [filter, setFilter] = React.useState<string | undefined>()
const { data } = useQuery({
queryKey: ['todos', filter],
// ⬇️ disabled as long as the filter is undefined or empty
queryFn: filter ? () => fetchTodos(filter) : skipToken,
})
return (
<div>
// 🚀 applying the filter will enable and execute the query
<FiltersForm onApply={setFilter} />
{data && <TodosTable data={data} />}
</div>
)
}
import { skipToken, useQuery } from '@tanstack/react-query'
function Todos() {
const [filter, setFilter] = React.useState<string | undefined>()
const { data } = useQuery({
queryKey: ['todos', filter],
// ⬇️ disabled as long as the filter is undefined or empty
queryFn: filter ? () => fetchTodos(filter) : skipToken,
})
return (
<div>
// 🚀 applying the filter will enable and execute the query
<FiltersForm onApply={setFilter} />
{data && <TodosTable data={data} />}
</div>
)
}