Skip to content

数据获取、缓存和重新验证

数据获取是任何应用程序的核心部分。本页介绍了如何在React和Next.js中获取、缓存和重新验证数据。

您可以以四种方式获取数据:

  1. 在服务器上,使用fetch
  2. 在服务器上,使用第三方库
  3. 在客户端上,通过路由处理器
  4. 在客户端上,使用第三方库

在服务器上使用fetch获取数据

Next.js扩展了原生的fetch Web API,允许您为服务器上的每个获取请求配置缓存重新验证行为。React扩展了fetch,以在渲染React组件树时自动记忆化获取请求。

您可以在服务器组件中、路由处理器中以及服务器操作中使用fetchasync/await

例如:

tsx
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>
}
jsx
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提供了在服务器组件中获取数据时可能需要的有用函数,如cookiesheaders。这些将导致路由动态渲染,因为它们依赖于请求时间信息。
  • 在路由处理器中,fetch请求不会被记忆化,因为路由处理器不是React组件树的一部分。
  • 服务器操作中,fetch请求不会被缓存(默认cache: no-store)。
  • 要在TypeScript的服务器组件中使用async/await,您需要使用TypeScript 5.1.3或更高版本和@types/react 18.2.8或更高版本。

缓存数据

缓存存储数据,以便在每次请求时不需要重新从数据源获取。

默认情况下,Next.js会在服务器上的数据缓存中自动缓存fetch返回的值。这意味着数据可以在构建时或请求时获取,缓存,并在每个数据请求上重用。

js
// 'force-cache'是默认设置,可以省略
fetch('https://...', { cache: 'force-cache' })

然而,有例外情况,当fetch请求在以下情况下不会被缓存:

什么是数据缓存?

数据缓存是一种持久的HTTP缓存。根据您的平台,缓存可以自动扩展,并可以跨多个区域共享

了解更多关于数据缓存的信息。

数据重新验证

数据重新验证是清除数据缓存并重新获取最新数据的过程。当您的数据发生变化,并且您希望确保显示最新信息时,这非常有用。

缓存数据可以通过两种方式重新验证:

  • 基于时间的重新验证:在一定时间后自动重新验证数据。这对于不经常变化且新鲜度不是非常关键的数据非常有用。
  • 按需重新验证:根据事件(例如表单提交)手动重新验证数据。按需重新验证可以使用基于标签或基于路径的方法一次性重新验证数据组。当您希望尽快显示最新数据时(例如,当您的无头CMS内容更新时),这非常有用。

基于时间的重新验证

要在定时间隔重新验证数据,您可以使用 fetchnext.revalidate 选项来设置资源的缓存生命周期(以秒为单位)。

js
fetch('https://...', { next: { revalidate: 3600 } })

或者,要重新验证路由段中的所有 fetch 请求,您可以使用 Segment Config Options

jsx
export const revalidate = 3600 // 每小时重新验证一次

如果您在静态渲染的路由中有多个 fetch 请求,并且每个请求的重新验证频率不同,则将使用最低时间用于所有请求。对于动态渲染的路由,每个 fetch 请求将独立重新验证。

了解更多关于 基于时间的重新验证

按需重新验证

可以通过路径 (revalidatePath) 或缓存标签 (revalidateTag) 按需重新验证数据,这可以在 Server ActionRoute Handler 中实现。

Next.js 有一个用于跨路由使 fetch 请求失效的缓存标签系统。

  1. 当使用 fetch 时,您可以选择使用一个或多个标签标记缓存条目。
  2. 然后,您可以调用 revalidateTag 来重新验证与该标签关联的所有条目。

例如,以下 fetch 请求添加了缓存标签 collection

tsx
export default async function Page() {
  const res = await fetch('https://...', { next: { tags: ['collection'] } })
  const data = await res.json()
  // ...
}
jsx
export default async function Page() {
  const res = await fetch('https://...', { next: { tags: ['collection'] } })
  const data = await res.json()
  // ...
}

然后,您可以通过在 Server Action 中调用 revalidateTag 来重新验证标记为 collection 的此 fetch 调用:

ts
'use server'

import { revalidateTag } from 'next/cache'

export default async function action() {
  revalidateTag('collection')
}
js
'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 请求在组件树中使用了 headerscookies 之后。
  • 使用了 const dynamic = 'force-dynamic' 路由段选项。
  • fetchCache 路由段选项被配置为默认跳过缓存。
  • fetch 请求使用了 AuthorizationCookie 头,并且在组件树中它的上方有一个未被缓存的请求。

个别 fetch 请求

要为个别 fetch 请求禁用缓存,您可以在 fetchcache 选项中设置为 'no-store'。这将在每次请求时动态获取数据。

js
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,这意味着数据将被缓存并每小时最多重新验证一次。
ts
import { cache } from 'react'

export const getItem = cache(async (id: string) => {
  const item = await db.item.findUnique({ id })
  return item
})
js
import { cache } from 'react'

export const getItem = cache(async (id) => {
  const item = await db.item.findUnique({ id })
  return item
})

尽管getItem函数被调用了两次,但只会向数据库发出一次查询。

tsx
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)
  // ...
}
jsx
import { getItem } from '@/utils/get-item'

export const revalidate = 3600 // 每小时最多重新验证一次数据

export default async function Layout({ params: { id } }) {
  const item = await getItem(id)
  // ...
}
tsx
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)
  // ...
}
jsx
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内部获取数据。

客户端使用第三方库获取数据

您也可以使用第三方库,如SWRTanStack Query,在客户端获取数据。这些库提供了自己的API,用于记忆请求、缓存、重新验证和修改数据。

须知

use是一个React函数,它接受并处理由函数返回的promise。在客户端组件中,目前不推荐fetch包装在use中,这可能会触发多次重新渲染。在React文档中了解更多关于use的信息。