数据获取、缓存和重新验证
数据获取是任何应用程序的核心部分。本页介绍了如何在React和Next.js中获取、缓存和重新验证数据。
您可以以四种方式获取数据:
在服务器上使用fetch
获取数据
Next.js扩展了原生的fetch
Web API,允许您为服务器上的每个获取请求配置缓存和重新验证行为。React扩展了fetch
,以在渲染React组件树时自动记忆化获取请求。
您可以在服务器组件中、路由处理器中以及服务器操作中使用fetch
与async
/await
。
例如:
async function getData() {
const res = await fetch('https://api.example.com/...')
// 返回值*不是*序列化的
// 您可以返回Date, Map, Set等
if (!res.ok) {
// 这将激活最近的`error.js`错误边界
throw new Error('Failed to fetch data')
}
return res.json()
}
export default async function Page() {
const data = await getData()
return <main></main>
}
async function getData() {
const res = await fetch('https://api.example.com/...')
// 返回值*不是*序列化的
// 您可以返回Date, Map, Set等
if (!res.ok) {
// 这将激活最近的`error.js`错误边界
throw new Error('Failed to fetch data')
}
return res.json()
}
export default async function Page() {
const data = await getData()
return <main></main>
}
须知:
缓存数据
缓存存储数据,以便在每次请求时不需要重新从数据源获取。
默认情况下,Next.js会在服务器上的数据缓存中自动缓存fetch
返回的值。这意味着数据可以在构建时或请求时获取,缓存,并在每个数据请求上重用。
// 'force-cache'是默认设置,可以省略
fetch('https://...', { cache: 'force-cache' })
然而,有例外情况,当fetch
请求在以下情况下不会被缓存:
什么是数据缓存?
数据缓存是一种持久的HTTP缓存。根据您的平台,缓存可以自动扩展,并可以跨多个区域共享。
了解更多关于数据缓存的信息。
数据重新验证
数据重新验证是清除数据缓存并重新获取最新数据的过程。当您的数据发生变化,并且您希望确保显示最新信息时,这非常有用。
缓存数据可以通过两种方式重新验证:
- 基于时间的重新验证:在一定时间后自动重新验证数据。这对于不经常变化且新鲜度不是非常关键的数据非常有用。
- 按需重新验证:根据事件(例如表单提交)手动重新验证数据。按需重新验证可以使用基于标签或基于路径的方法一次性重新验证数据组。当您希望尽快显示最新数据时(例如,当您的无头CMS内容更新时),这非常有用。
基于时间的重新验证
要在定时间隔重新验证数据,您可以使用 fetch
的 next.revalidate
选项来设置资源的缓存生命周期(以秒为单位)。
fetch('https://...', { next: { revalidate: 3600 } })
或者,要重新验证路由段中的所有 fetch
请求,您可以使用 Segment Config Options。
export const revalidate = 3600 // 每小时重新验证一次
如果您在静态渲染的路由中有多个 fetch
请求,并且每个请求的重新验证频率不同,则将使用最低时间用于所有请求。对于动态渲染的路由,每个 fetch
请求将独立重新验证。
了解更多关于 基于时间的重新验证。
按需重新验证
可以通过路径 (revalidatePath
) 或缓存标签 (revalidateTag
) 按需重新验证数据,这可以在 Server Action 或 Route Handler 中实现。
Next.js 有一个用于跨路由使 fetch
请求失效的缓存标签系统。
- 当使用
fetch
时,您可以选择使用一个或多个标签标记缓存条目。 - 然后,您可以调用
revalidateTag
来重新验证与该标签关联的所有条目。
例如,以下 fetch
请求添加了缓存标签 collection
:
export default async function Page() {
const res = await fetch('https://...', { next: { tags: ['collection'] } })
const data = await res.json()
// ...
}
export default async function Page() {
const res = await fetch('https://...', { next: { tags: ['collection'] } })
const data = await res.json()
// ...
}
然后,您可以通过在 Server Action 中调用 revalidateTag
来重新验证标记为 collection
的此 fetch
调用:
'use server'
import { revalidateTag } from 'next/cache'
export default async function action() {
revalidateTag('collection')
}
'use server'
import { revalidateTag } from 'next/cache'
export default async function action() {
revalidateTag('collection')
}
了解更多关于 按需重新验证。
错误处理和重新验证
如果在尝试重新验证数据时抛出错误,最后成功生成的数据将继续从缓存中提供。在下一次随后的请求中,Next.js 将重试重新验证数据。
数据缓存的禁用
如果满足以下条件,fetch
请求将不被缓存:
- 在
fetch
请求中添加了cache: 'no-store'
。 - 在个别
fetch
请求中添加了revalidate: 0
选项。 fetch
请求位于使用POST
方法的路由处理器中。fetch
请求在组件树中使用了headers
或cookies
之后。- 使用了
const dynamic = 'force-dynamic'
路由段选项。 fetchCache
路由段选项被配置为默认跳过缓存。fetch
请求使用了Authorization
或Cookie
头,并且在组件树中它的上方有一个未被缓存的请求。
个别 fetch
请求
要为个别 fetch
请求禁用缓存,您可以在 fetch
的 cache
选项中设置为 'no-store'
。这将在每次请求时动态获取数据。
fetch('https://...', { cache: 'no-store' })
在 fetch
API 参考 中查看所有可用的 cache
选项。
多个 fetch
请求
如果您在路由段(例如布局或页面)中有多个 fetch
请求,您可以使用 Segment Config Options 配置该段中所有数据请求的缓存行为。
然而,我们建议个别配置每个 fetch
请求的缓存行为。这可以让您更细致地控制缓存行为。
Fetching data on the Server with third-party libraries
在使用第三方库时,如果该库不支持或不公开fetch
(例如数据库、CMS或ORM客户端),您可以使用Route Segment Config Option和React的cache
函数配置这些请求的缓存和重新验证行为。
数据是否被缓存将取决于路由段是静态还是动态渲染。如果段是静态的(默认),请求的输出将被缓存,并作为路由段的一部分进行重新验证。如果段是动态的,请求的输出将不会被缓存,并且在每次渲染段时都会重新获取。
您也可以使用实验性的unstable_cache
API。
示例
在下面的示例中:
- 使用React的
cache
函数来记忆化数据请求。 - 在布局和页面段中将
revalidate
选项设置为3600
,这意味着数据将被缓存并每小时最多重新验证一次。
import { cache } from 'react'
export const getItem = cache(async (id: string) => {
const item = await db.item.findUnique({ id })
return item
})
import { cache } from 'react'
export const getItem = cache(async (id) => {
const item = await db.item.findUnique({ id })
return item
})
尽管getItem
函数被调用了两次,但只会向数据库发出一次查询。
import { getItem } from '@/utils/get-item'
export const revalidate = 3600 // 每小时最多重新验证一次数据
export default async function Layout({
params: { id },
}: {
params: { id: string }
}) {
const item = await getItem(id)
// ...
}
import { getItem } from '@/utils/get-item'
export const revalidate = 3600 // 每小时最多重新验证一次数据
export default async function Layout({ params: { id } }) {
const item = await getItem(id)
// ...
}
import { getItem } from '@/utils/get-item'
export const revalidate = 3600 // 每小时最多重新验证一次数据
export default async function Page({
params: { id },
}: {
params: { id: string }
}) {
const item = await getItem(id)
// ...
}
import { getItem } from '@/utils/get-item'
export const revalidate = 3600 // 每小时最多重新验证一次数据
export default async function Page({ params: { id } }) {
const item = await getItem(id)
// ...
}
Fetching Data on the Client with Route Handlers
如果您需要在客户端组件中获取数据,您可以从客户端调用Route Handler。Route Handlers在服务器上执行并将数据返回给客户端。这在您不想向客户端公开敏感信息时非常有用,例如API令牌。
请参阅Route Handler文档中的示例。
Server Components和Route Handlers
由于Server Components在服务器上渲染,您不需要从Server Component调用Route Handler来获取数据。相反,您可以直接在Server Component内部获取数据。
客户端使用第三方库获取数据
您也可以使用第三方库,如SWR或TanStack Query,在客户端获取数据。这些库提供了自己的API,用于记忆请求、缓存、重新验证和修改数据。
须知:
use
是一个React函数,它接受并处理由函数返回的promise。在客户端组件中,目前不推荐将fetch
包装在use
中,这可能会触发多次重新渲染。在React文档中了解更多关于use
的信息。