跳到内容
API 参考函数getStaticProps

getStaticProps

导出名为 getStaticProps 的函数将在构建时使用该函数返回的 props 预渲染页面

pages/index.tsx
import type { InferGetStaticPropsType, GetStaticProps } from 'next'
 
type Repo = {
  name: string
  stargazers_count: number
}
 
export const getStaticProps = (async (context) => {
  const res = await fetch('https://api.github.com/repos/vercel/next.js')
  const repo = await res.json()
  return { props: { repo } }
}) satisfies GetStaticProps<{
  repo: Repo
}>
 
export default function Page({
  repo,
}: InferGetStaticPropsType<typeof getStaticProps>) {
  return repo.stargazers_count
}

你可以在顶层作用域中导入模块,以在 getStaticProps 中使用。使用的导入将不会为客户端捆绑。这意味着你可以直接在 getStaticProps 中编写服务器端代码,包括从数据库中获取数据。

Context 参数

context 参数是一个包含以下键的对象

名称描述
params包含使用动态路由的页面的路由参数。例如,如果页面名称是 [id].js,则 params 将看起来像 { id: ... }。你应该将此与 getStaticPaths 一起使用,我们稍后会解释。
preview(已弃用,请使用 draftMode)如果页面处于预览模式,则 previewtrue,否则为 false
previewData(已弃用,请使用 draftMode)由 setPreviewData 设置的预览数据。
draftMode如果页面处于草稿模式,则 draftModetrue,否则为 false
locale包含活动的区域设置(如果已启用)。
locales包含所有支持的区域设置(如果已启用)。
defaultLocale包含配置的默认区域设置(如果已启用)。
revalidateReason提供函数被调用的原因。可以是以下之一:“build”(构建时运行)、“stale”(重新验证周期已过期,或在开发模式下运行)、“on-demand”(通过按需重新验证触发)

getStaticProps 返回值

getStaticProps 函数应返回一个对象,其中包含 propsredirectnotFound,后跟一个可选的 revalidate 属性。

props

props 对象是一个键值对,其中每个值都由页面组件接收。它应该是一个可序列化的对象,以便可以与 JSON.stringify序列化任何传递的 props。

export async function getStaticProps(context) {
  return {
    props: { message: `Next.js is awesome` }, // will be passed to the page component as props
  }
}

revalidate

revalidate 属性是以秒为单位的时间,在此时间之后可以发生页面重新生成(默认为 false 或不重新验证)。

// This function gets called at build time on server-side.
// It may be called again, on a serverless function, if
// revalidation is enabled and a new request comes in
export async function getStaticProps() {
  const res = await fetch('https://.../posts')
  const posts = await res.json()
 
  return {
    props: {
      posts,
    },
    // Next.js will attempt to re-generate the page:
    // - When a request comes in
    // - At most once every 10 seconds
    revalidate: 10, // In seconds
  }
}

了解更多关于 增量静态再生

利用 ISR 的页面的缓存状态可以通过读取 x-nextjs-cache 响应头的值来确定。可能的值如下:

  • MISS - 路径不在缓存中(最多发生一次,在首次访问时)
  • STALE - 路径在缓存中,但超过了重新验证时间,因此将在后台更新
  • HIT - 路径在缓存中,并且未超过重新验证时间

notFound

notFound 布尔值允许页面返回 404 状态和 404 页面。 使用 notFound: true,即使之前成功生成了页面,该页面也将返回 404。 这旨在支持用户生成的内容被其作者删除等用例。 注意,notFound 遵循与 此处描述相同的 revalidate 行为。

export async function getStaticProps(context) {
  const res = await fetch(`https://.../data`)
  const data = await res.json()
 
  if (!data) {
    return {
      notFound: true,
    }
  }
 
  return {
    props: { data }, // will be passed to the page component as props
  }
}

须知:对于 fallback: false 模式,不需要 notFound,因为只有从 getStaticPaths 返回的路径才会被预渲染。

redirect

redirect 对象允许重定向到内部或外部资源。它应该匹配 { destination: string, permanent: boolean } 的形状。

在某些极少数情况下,您可能需要为较旧的 HTTP 客户端分配自定义状态代码,以便正确重定向。在这些情况下,您可以使用 statusCode 属性来代替 permanent 属性,但不能同时使用两者。您还可以设置类似于 next.config.js 中重定向的 basePath: false

export async function getStaticProps(context) {
  const res = await fetch(`https://...`)
  const data = await res.json()
 
  if (!data) {
    return {
      redirect: {
        destination: '/',
        permanent: false,
        // statusCode: 301
      },
    }
  }
 
  return {
    props: { data }, // will be passed to the page component as props
  }
}

如果重定向在构建时已知,则应将其添加到 next.config.js 中。

读取文件:使用 process.cwd()

文件可以直接从 getStaticProps 中的文件系统读取。

为了做到这一点,您必须获取文件的完整路径。

由于 Next.js 将您的代码编译到单独的目录中,因此您不能使用 __dirname,因为它返回的路径将与 Pages Router 不同。

相反,您可以使用 process.cwd(),它会为您提供 Next.js 正在执行的目录。

import { promises as fs } from 'fs'
import path from 'path'
 
// posts will be populated at build time by getStaticProps()
function Blog({ posts }) {
  return (
    <ul>
      {posts.map((post) => (
        <li>
          <h3>{post.filename}</h3>
          <p>{post.content}</p>
        </li>
      ))}
    </ul>
  )
}
 
// This function gets called at build time on server-side.
// It won't be called on client-side, so you can even do
// direct database queries.
export async function getStaticProps() {
  const postsDirectory = path.join(process.cwd(), 'posts')
  const filenames = await fs.readdir(postsDirectory)
 
  const posts = filenames.map(async (filename) => {
    const filePath = path.join(postsDirectory, filename)
    const fileContents = await fs.readFile(filePath, 'utf8')
 
    // Generally you would parse/transform the contents
    // For example you can transform markdown to HTML here
 
    return {
      filename,
      content: fileContents,
    }
  })
  // By returning { props: { posts } }, the Blog component
  // will receive `posts` as a prop at build time
  return {
    props: {
      posts: await Promise.all(posts),
    },
  }
}
 
export default Blog

版本历史

版本变更
v13.4.0App Router 现在已稳定,并简化了数据获取
v12.2.0按需增量静态生成 已稳定。
v12.1.0添加了 按需增量静态生成(beta 版)。
v10.0.0添加了 localelocalesdefaultLocalenotFound 选项。
v10.0.0添加了 fallback: 'blocking' 返回选项。
v9.5.0稳定的 增量静态生成
v9.3.0引入了 getStaticProps