getStaticPaths
当从使用动态路由的页面导出名为 getStaticPaths
的函数时,Next.js 将静态预渲染 getStaticPaths
指定的所有路径。
import type {
InferGetStaticPropsType,
GetStaticProps,
GetStaticPaths,
} from 'next'
type Repo = {
name: string
stargazers_count: number
}
export const getStaticPaths = (async () => {
return {
paths: [
{
params: {
name: 'next.js',
},
}, // See the "paths" section below
],
fallback: true, // false or "blocking"
}
}) satisfies GetStaticPaths
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
}
getStaticPaths 返回值
getStaticPaths
函数应返回一个对象,其中包含以下**必需**属性
paths
paths
键确定哪些路径将被预渲染。例如,假设你有一个使用动态路由的页面,名为 pages/posts/[id].js
。如果你从此页面导出 getStaticPaths
并为 paths
返回以下内容
return {
paths: [
{ params: { id: '1' }},
{
params: { id: '2' },
// with i18n configured the locale for the path can be returned as well
locale: "en",
},
],
fallback: ...
}
那么,Next.js 将在 next build
期间使用 pages/posts/[id].js
中的页面组件静态生成 /posts/1
和 /posts/2
。
每个 params
对象的值必须与页面名称中使用的参数匹配
- 如果页面名称是
pages/posts/[postId]/[commentId]
,则params
应包含postId
和commentId
。 - 如果页面名称使用类似
pages/[...slug]
的捕获所有路由,则params
应包含slug
(它是一个数组)。如果此数组是['hello', 'world']
,那么 Next.js 将在/hello/world
静态生成页面。 - 如果页面使用可选的捕获所有路由,请使用
null
、[]
、undefined
或false
来渲染最根部的路由。例如,如果你为pages/[[...slug]]
提供slug: false
,Next.js 将静态生成页面/
。
params
字符串是**区分大小写**的,理想情况下应该被规范化,以确保路径被正确生成。例如,如果为参数返回 WoRLD
,则只有当实际访问的路径是 WoRLD
时才会匹配,而不是 world
或 World
。
除了 params
对象之外,当配置了 i18n 时,还可以返回 locale
字段,该字段配置要生成的路径的语言环境。
fallback: false
如果 fallback
为 false
,则任何未被 getStaticPaths
返回的路径都将导致 **404 页面**。
当运行 next build
时,Next.js 将检查 getStaticPaths
是否返回了 fallback: false
,然后它将**仅**构建 getStaticPaths
返回的路径。如果你需要创建的路径数量较少,或者不经常添加新的页面数据,则此选项很有用。如果你发现需要添加更多路径,并且你设置了 fallback: false
,则需要再次运行 next build
,以便可以生成新路径。
以下示例预渲染每个页面一篇博客文章,页面名为 pages/posts/[id].js
。博客文章列表将从 CMS 获取并由 getStaticPaths
返回。然后,对于每个页面,它使用 getStaticProps
从 CMS 获取帖子数据。
function Post({ post }) {
// Render post...
}
// This function gets called at build time
export async function getStaticPaths() {
// Call an external API endpoint to get posts
const res = await fetch('https://.../posts')
const posts = await res.json()
// Get the paths we want to pre-render based on posts
const paths = posts.map((post) => ({
params: { id: post.id },
}))
// We'll pre-render only these paths at build time.
// { fallback: false } means other routes should 404.
return { paths, fallback: false }
}
// This also gets called at build time
export async function getStaticProps({ params }) {
// params contains the post `id`.
// If the route is like /posts/1, then params.id is 1
const res = await fetch(`https://.../posts/${params.id}`)
const post = await res.json()
// Pass post data to the page via props
return { props: { post } }
}
export default Post
fallback: true
示例
如果 fallback
为 true
,则 getStaticProps
的行为会发生以下变化
- 从
getStaticPaths
返回的路径将在构建时通过getStaticProps
渲染为 HTML。 - 在构建时未生成的路径**不会**导致 404 页面。相反,Next.js 将在首次请求此类路径时提供页面的 “fallback” 版本。Web 爬虫(如 Google)不会收到 fallback,而是路径的行为将与
fallback: 'blocking'
中一样。 - 当通过
next/link
或next/router
(客户端)导航到fallback: true
的页面时,Next.js **不会**提供 fallback,而是页面的行为将与fallback: 'blocking'
相同。 - 在后台,Next.js 将静态生成请求的路径 HTML 和 JSON。这包括运行
getStaticProps
。 - 完成后,浏览器会收到生成的路径的 JSON。这将用于自动渲染具有所需 props 的页面。从用户的角度来看,页面将从 fallback 页面切换到完整页面。
- 同时,Next.js 将此路径添加到预渲染页面列表。后续对同一路径的请求将提供生成的页面,就像构建时预渲染的其他页面一样。
须知:当使用
output: 'export'
时,不支持fallback: true
。
何时 fallback: true
有用?
如果你的应用有大量依赖数据的静态页面(例如,一个非常大的电子商务网站),则 fallback: true
非常有用。如果你想预渲染所有产品页面,构建将花费很长时间。
相反,你可以静态生成一小部分页面,并对其余页面使用 fallback: true
。当有人请求尚未生成的页面时,用户将看到带有加载指示器或骨架组件的页面。
不久之后,getStaticProps
完成,页面将使用请求的数据进行渲染。从现在开始,每个请求同一页面的人都将获得静态预渲染的页面。
这确保用户始终获得快速的体验,同时保持快速构建和静态生成的好处。
fallback: true
**不会**更新生成的页面,要更新生成的页面,请查看 增量静态再生。
fallback: 'blocking'
如果 fallback
为 'blocking'
,则未被 getStaticPaths
返回的新路径将等待 HTML 生成,这与 SSR 相同(因此称为 *blocking*),然后会被缓存以供将来的请求使用,因此每个路径仅发生一次。
getStaticProps
的行为如下
- 从
getStaticPaths
返回的路径将在构建时通过getStaticProps
渲染为 HTML。 - 在构建时未生成的路径**不会**导致 404 页面。相反,Next.js 将在首次请求时执行 SSR 并返回生成的 HTML。
- 完成后,浏览器会收到生成的路径的 HTML。从用户的角度来看,它将从 “浏览器正在请求页面” 过渡到 “完整页面已加载”。没有加载/fallback 状态的闪烁。
- 同时,Next.js 将此路径添加到预渲染页面列表。后续对同一路径的请求将提供生成的页面,就像构建时预渲染的其他页面一样。
默认情况下,fallback: 'blocking'
**不会**更新生成的页面。要更新生成的页面,请将 增量静态再生 与 fallback: 'blocking'
结合使用。
须知:当使用
output: 'export'
时,不支持fallback: 'blocking'
。
Fallback 页面
在页面的 “fallback” 版本中
- 页面的 props 将为空。
- 使用 router,你可以检测 fallback 是否正在渲染,
router.isFallback
将为true
。
以下示例展示了如何使用 isFallback
import { useRouter } from 'next/router'
function Post({ post }) {
const router = useRouter()
// If the page is not yet generated, this will be displayed
// initially until getStaticProps() finishes running
if (router.isFallback) {
return <div>Loading...</div>
}
// Render post...
}
// This function gets called at build time
export async function getStaticPaths() {
return {
// Only `/posts/1` and `/posts/2` are generated at build time
paths: [{ params: { id: '1' } }, { params: { id: '2' } }],
// Enable statically generating additional pages
// For example: `/posts/3`
fallback: true,
}
}
// This also gets called at build time
export async function getStaticProps({ params }) {
// params contains the post `id`.
// If the route is like /posts/1, then params.id is 1
const res = await fetch(`https://.../posts/${params.id}`)
const post = await res.json()
// Pass post data to the page via props
return {
props: { post },
// Re-generate the post at most once per second
// if a request comes in
revalidate: 1,
}
}
export default Post
版本历史
版本 | 变更 |
---|---|
v13.4.0 | App Router 现在已稳定,并简化了数据获取,包括 generateStaticParams() |
v12.2.0 | 按需增量静态再生 已稳定。 |
v12.1.0 | 添加了 按需增量静态再生(beta 版)。 |
v9.5.0 | 稳定的 增量静态再生 |
v9.3.0 | 引入了 getStaticPaths 。 |
这有帮助吗?