Skip to main content

无限查询 Infinite Queries

通过"加载更多"来附加更多的数据到现有数据集,或者是通过"无限滚动"来呈现列表,这些都是非常常见的 UI 模式。 正巧,React Query 支持一个有用的useQuery版本,称为useInfiniteQuery,很适合用来查询这些类型的数据。

使用useInfiniteQuery时,需要注意一些不同之处:

  • data现在是一个包含无限查询数据的对象
    • data.pages数组包含已获取的分页页面
    • data.pageParams数组包含用于获取分页页面的参数
  • 现在可以使用fetchNextPagefetchPreviousPage函数
  • getNextPageParamgetPreviousPageParam选项可用于确定是否有更多数据要加载和或用来获取它所需的信息。此信息作为查询函数中的附加参数提供(在调用fetchNextPagefetchPreviousPage函数时可以有选择地覆盖此信息)
  • hasNextPage布尔值现在可用。如果getNextPageParam返回的值不是undefined的,则为true
  • hasPreviousPage布尔值现在可用。如果getPreviousPageParam返回的值不是undefined的,则为true
  • isFetchingNextPageisFetchingPreviousPage布尔值来区分后台刷新状态和加载更多状态

注意:在查询中使用initialDataselect参数时,请确保在重组(restructure)数据时仍包含data.pagesdata.pageParams属性,否则查询返回时将覆盖你的更改!

例子

假设我们有一个基于游标索引的,且每次返回三项projects的 API,以及一个可用于获取下一组项目的游标:

fetch("/api/projects?cursor=0");
// { data: [...], nextCursor: 3}
fetch("/api/projects?cursor=3");
// { data: [...], nextCursor: 6}
fetch("/api/projects?cursor=6");
// { data: [...], nextCursor: 9}
fetch("/api/projects?cursor=9");
// { data: [...] }

有了这些信息,我们可以通过以下方式创建"加载更多"的 UI:

  • 默认情况下,等待useInfiniteQuery请求第一组数据
  • 获取getNextPageParam中返回的用于下一个查询的信息
  • 调用fetchNextPage函数

注意:请勿使用参数调用fetchNextPage,这很重要。除非你希望它们覆盖从getNextPageParam函数返回的pageParam数据。 例如,不要这样做:<button onClick={fetchNextPage} />,因为这会将onClick的事件对象当作一个参数给fetchNextPage函数。

import { useInfiniteQuery } from "@tanstack/react-query";

function Projects() {
const fetchProjects = async ({ pageParam = 0 }) => {
const res = await fetch("/api/projects?cursor=" + pageParam);
return res.json();
};

const {
data,
error,
fetchNextPage,
hasNextPage,
isFetching,
isFetchingNextPage,
status,
} = useInfiniteQuery({
queryKey: ["projects"],
queryFn: fetchProjects,
getNextPageParam: (lastPage, pages) => lastPage.nextCursor,
});

return status === "loading" ? (
<p>Loading...</p>
) : status === "error" ? (
<p>Error: {error.message}</p>
) : (
<>
{data.pages.map((group, i) => (
<React.Fragment key={i}>
{group.projects.map((project) => (
<p key={project.id}>{project.name}</p>
))}
</React.Fragment>
))}
<div>
<button
onClick={() => fetchNextPage()}
disabled={!hasNextPage || isFetchingNextPage}
>
{isFetchingNextPage
? "Loading more..."
: hasNextPage
? "Load More"
: "Nothing more to load"}
</button>
</div>
<div>{isFetching && !isFetchingNextPage ? "Fetching..." : null}</div>
</>
);
}

当无限查询需要重新获取时会发生什么

当无限查询变得陈旧(stale)且需要重新获取时,将从第一个查询开始,按照对应顺序获取每个组。 这样可以确保即使基础数据发生了修改,我们也不会使用过时的游标,也不会得到重复的记录或跳过记录。 如果从 queryCache 中删除了无限查询的结果,则分页将在初始状态下重新启动,仅请求初始的第一个分组(initial group)。

重新获取页面

如果你只想主动重新获取某一部分页面,可以通过传递给refetch(从useInfiniteQuery返回的)一个refetchPage函数来做到

const { refetch } = useInfiniteQuery({
queryKey: ["projects"],
queryFn: fetchProjects,
getNextPageParam: (lastPage, pages) => lastPage.nextCursor,
});

// 只重新获取 index===0 的页面
refetch({ refetchPage: (page, index) => index === 0 });

你也可以将该方法作为第二个参数(queryFilters)传递给 queryClient.refetchQueries, queryClient.invalidateQueries 或者 queryClient.resetQueries 来做到同样的效果

函数签名

  • refetchPage: (page: TData, index: number, allPages: TData[]) => boolean

该函数对每个页面执行一次,只有返回true的页面才会被重新获取。

如果我需要将自定义信息传递给查询函数怎么办

默认情况下,从getNextPageParam返回的变量将提供给查询函数,但是在某些情况下,你可能希望重写它。 你可以将自定义变量传递给fetchNextPage函数,该函数将覆盖默认变量,如下所示:

function Projects() {
const fetchProjects = ({ pageParam = 0 }) =>
fetch("/api/projects?cursor=" + pageParam);

const {
status,
data,
isFetching,
isFetchingNextPage,
fetchNextPage,
hasNextPage,
} = useInfiniteQuery({
queryKey: ["projects"],
queryFn: fetchProjects,
getNextPageParam: (lastPage, pages) => lastPage.nextCursor,
});

// 传递你自己的页面参数
const skipToCursor50 = () => fetchNextPage({ pageParam: 50 });
}

如果我想实现双向无限列表怎么办

双向列表可以通过使用getPreviousPageParamfetchPreviousPagehasPreviousPageisFetchingPreviousPage属性和函数来实现。

useInfiniteQuery({
queryKey: ["projects"],
queryFn: fetchProjects,
getNextPageParam: (lastPage, pages) => lastPage.nextCursor,
getPreviousPageParam: (firstPage, pages) => firstPage.prevCursor,
});

如果我想以相反的顺序显示页面呢

有时你可能想以相反的顺序显示页面。在这种情况下,可以使用select选项:

useInfiniteQuery({
queryKey: ["projects"],
queryFn: fetchProjects,
select: (data) => ({
pages: [...data.pages].reverse(),
pageParams: [...data.pageParams].reverse(),
}),
});

如果要手动更新无限查询该怎么办

手动删除第一页:

queryClient.setQueryData(["projects"], (data) => ({
pages: data.pages.slice(1),
pageParams: data.pageParams.slice(1),
}));

从单个页面中手动删除某一个值:

const newPagesArray =
oldPagesArray?.pages.map((page) =>
page.filter((val) => val.id !== updatedId)
) ?? [];

queryClient.setQueryData(["projects"], (data) => ({
pages: newPagesArray,
pageParams: data.pageParams,
}));

确保分页页面和pageParams保持相同的数据结构!