初始化的查询数据 Initial Query Data
在需要查询之前,有很多方法可以向缓存提供初始数据:
- 声明式的:
- 向查询提供
initialData
以预填充其缓存(如果为空)
- 向查询提供
- 命令式的:
使用initialData
预先填充查询
有时候,你可能已经在应用中其他地方获得了查询的初始数据,并且可以直接将其提供给你的查询。
在这种情况下,则可以使用config.initialData
选项设置查询的初始数据,并跳过初始加载状态!
重要说明:
initialData
保留在缓存中,因此不建议为此选项提供占位符,部分或不完整的数据,如有必要,应使用placeholderData
const result = useQuery({
queryKey: ["todos"],
queryFn: () => fetch("/todos"),
initialData: initialTodos,
});
staleTime
和 initialDataUpdatedAt
默认情况下,initialData
被视为完全新鲜的数据,就像它刚刚被获取一样。这也意味着它将影响staleTime
对它的解释。
如果为查询观察者(query observer)配置了
initialData
且没有staleTime
(默认的staleTime: 0
),则该查询在挂载时将立即重新获取:// 将立即显示 initialTodos,但在挂载后也将立即重新获取todos
const result = useQuery({
queryKey: ["todos"],
queryFn: () => fetch("/todos"),
initialData: initialTodos,
});如果为查询观察者(query observer)配置了
initialData
且staleTime
为1000
ms,则在相同的时间量内数据将被视为最新数据,就好像它是刚从查询中获取的一样// 立即显示 initialTodos,但是直到 1000 ms 之后遇到另一个交互事件时才重新获取数据
const result = useQuery({
queryKey: ["todos"],
queryFn: () => fetch("/todos"),
initialData: initialTodos,
staleTime: 1000,
});那么,如果你的
initialData
并不完全新鲜怎么办?这就引出了最后一个--实际上也是最为准确的--名为initialDataUpdatedAt
的配置项。该选项允许你传递一个Number
类型的 JS 时间戳(以毫秒为单位,如Date.now()
),以确定initialData
上次更新的时间。请注意,如果是 unix 时间戳,则需要将其乘以 1000,以将其转换为 JS 时间戳// 立即显示 initialTodos,但是直到 1000 ms 之后遇到另一个交互事件时才重新获取数据
const result = useQuery({
queryKey: ["todos"],
queryFn: () => fetch("/todos"),
initialData: initialTodos,
staleTime: 60 * 1000,
initialData: initialTodos,
staleTime: 60 * 1000, // 1 minute
// 这可能是10秒前或10分钟前
initialDataUpdatedAt: initialTodosUpdatedTimestamp, // eg. 1608412420052
});此选项允许将 staleTime 用于其"原始"用途,以确定数据需要有多新鲜,同时还允许:如果
initialData
早于staleTime
,则在挂载时重新获取数据。在上面的示例中,我们的数据需要在 1 分钟内刷新,我们可以在initialData
最后一次更新时提示查询,以便查询自己决定是否需要重新获取数据。
如果你希望将数据视为预取数据,建议你使用
prefetchQuery
或fetchQuery
API 来预先填充缓存,从而独立于initialData
配置staleTime
。
来自函数的初始数据
如果访问查询的初始数据的过程很繁琐,或者只是不想在每个渲染上执行某些操作,则可以传递一个函数作为initialData
值。这个函数只在初始化查询时执行一次,从而节省宝贵的 RAM 和或 CPU。
const result = useQuery({
queryKey: ["todos"],
queryFn: () => fetch("/todos"),
initialData: () => getExpensiveTodos(),
});
来自缓存的初始数据
在某些情况下,你可能希望能够从另一个查询的缓存结果中为查询提供初始数据。一个很好的例子是,从一个 todos list 查询中搜索一个单独的 todo 项,然后使用该项对应的已被缓存的数据,来作为单独的 todo 查询的初始数据:
const result = useQuery({
queryKey: ["todo", todoId],
queryFn: () => fetch("/todos"),
initialData: () => {
// 将 `todos` 查询中的某个 todo 用作此 todos 查询的初始数据
return queryClient.getQueryData(["todos"])?.find((d) => d.id === todoId);
},
});
来自配置InitialDataUpdatedAt
的缓存的初始数据
从缓存中获取初始数据意味着,哪怕使用了initialData
,用来查询初始数据的源查询还是有可能很旧。建议不要手动设置staleTime
来阻止查询被立即重新获取,而应该将源查询的dataUpdatedAt
传给initialDataUpdatedAt
参数。
这为查询实例提供了(确定是否以及何时需要重新获取查询)所需的所有信息,而不管是否提供了初始数据。
const result = useQuery({
queryKey: ["todos", todoId],
queryFn: () => fetch(`/todos/${todoId}`),
initialData: () =>
queryClient.getQueryData(["todos"])?.find((d) => d.id === todoId),
initialDataUpdatedAt: () =>
queryClient.getQueryState(["todos"])?.dataUpdatedAt,
});
来自缓存的有条件的初始数据
如果用来查找初始数据的源查询过旧,则可能根本不会考虑使用缓存的数据,而只是想从服务器中获取数据。
为了使此决定更容易,你可以改用queryClient.getQueryState
方法来获取关于源查询的更多信息,如state.dataUpdatedAt
时间戳。你可以以此确定查询是否足够"新鲜"以满足你的需求:
const result = useQuery({
queryKey: ["todo", todoId],
queryFn: () => fetch(`/todos/${todoId}`),
initialData: () => {
// 获取查询状态
const state = queryClient.getQueryState(["todos"]);
// 如果查询存在并且数据"老得"不超过10秒...
if (state && Date.now() - state.dataUpdatedAt <= 10 * 1000) {
// 返回单个todo
return state.data.find((d) => d.id === todoId);
}
// 否则,返回undefined并让它从硬加载状态获取!
},
});
延伸阅读
如果对于Initial Data
和Placeholder Data
有困惑的话,请参考此社区内容(英文)