- Java锁的逻辑(结合对象头和ObjectMonitor)
- 还在用饼状图?来瞧瞧这些炫酷的百分比可视化新图形(附代码实现)⛵
- 自动注册实体类到EntityFrameworkCore上下文,并适配ABP及ABPVNext
- 基于Sklearn机器学习代码实战
😂 好久前写了关于 getStaticProps 和 getStaticPaths 的内容,然而半年过去了源码解析就一直忘记了,不久前有人提醒才想起来,补下坑.
本文主要是解读下 getStaticProps 、 getStaticPaths 相关的源码,不了解这两个 API 的建议先看下之前的文章再看。👀 。
首先 getStaticProps 是应用于 SSG 场景,我们先看下 packages/next/server/render.tsx 中相关的代码:
const isSSG = !!getStaticProps;
const pageIsDynamic = isDynamicRoute(pathname);
if (isSSG && !isFallback) {
let data: UnwrapPromise<ReturnType<GetStaticProps>>;
try {
data = await getStaticProps!({
...(pageIsDynamic ? { params: query as ParsedUrlQuery } : undefined),
...(isPreview ? { preview: true, previewData: previewData } : undefined),
locales: renderOpts.locales,
locale: renderOpts.locale,
defaultLocale: renderOpts.defaultLocale
});
} catch (staticPropsError: any) {
// ....
}
// ...
}
isFallback 可以先不管。可以看到 getStaticProps 同样可以为异步函数,而是否为 SSG 就是由是否存在 getStaticProps 函数来决定的, SSG 场景下的 pageIsDynamic 则必须配合 getStaticPaths 使用,可以看到 getStaticProps 会接收几个参数:
params
是在动态页面的路由参数 previewData
和 preview: preview
模式的相关数据 locales, locale
和 defaultLocale
多语言相关参数 执行完成后 getStaticProps 的返回值会被放入 pageProps 中.
再看看 invalidKeys 相关部分,除了 revalidate 、 props 、 redirect 和 notFound 外别的属性都会被视为非法.
const invalidKeys = Object.keys(data).filter(
key => key !== 'revalidate' && key !== 'props' && key !== 'redirect' && key !== 'notFound'
);
if (invalidKeys.includes('unstable_revalidate')) {
throw new Error(UNSTABLE_REVALIDATE_RENAME_ERROR);
}
if (invalidKeys.length) {
throw new Error(invalidKeysMsg('getStaticProps', invalidKeys));
}
然后还有关于 notFound 和 redirect 的处理:
if ('notFound' in data && data.notFound) {
if (pathname === '/404') {
throw new Error(`The /404 page can not return notFound in "getStaticProps", please remove it to continue!`);
}
(renderOpts as any).isNotFound = true;
}
if ('redirect' in data && data.redirect && typeof data.redirect === 'object') {
checkRedirectValues(data.redirect as Redirect, req, 'getStaticProps');
if (isBuildTimeSSG) {
throw new Error(
`\`redirect\` can not be returned from getStaticProps during prerendering (${req.url})\n` +
`See more info here: https://nextjs.org/docs/messages/gsp-redirect-during-prerender`
);
}
(data as any).props = {
__N_REDIRECT: data.redirect.destination,
__N_REDIRECT_STATUS: getRedirectStatus(data.redirect)
};
if (typeof data.redirect.basePath !== 'undefined') {
(data as any).props.__N_REDIRECT_BASE_PATH = data.redirect.basePath;
}
(renderOpts as any).isRedirect = true;
}
notFound 会使用 renderOpts.isNotFound 来标识,而 redirect 则会在 props 中通过 __N_REDIRECT 相关的参数来进行标识.
当然这里省略很多的校验,比如 getStaticProps 和 getServerSideProps 冲突、 getStaticPaths 的检查、 notFound 和 redirect 不能同时存在等.
props.pageProps = Object.assign({}, props.pageProps, 'props' in data ? data.props : undefined);
然后其中还包含了一部分与 revalidate 相关的内容,主要是一些检测和值的处理,主要与 ISR 相关的此处先跳过.
getStaticPaths 的相关的调用源码主要在 packages/next/build/utils.ts 文件中的 buildStaticPaths 中, buildStaticPaths 会在两个时候被调用,一个是 next.js 构建的时候,第二个是 next.js 的 devServer 中。在 next.js 遇到动态路由时,会按照 buildStaticPaths 和 getStaticProps 来决定是否启用 SSG 模式,启用则会调用 buildStaticPaths 获取该动态路由所对应的需要构建的所有静态页面.
if (getStaticPaths) {
staticPathsResult = await getStaticPaths({ locales, defaultLocale });
}
if (!staticPathsResult || typeof staticPathsResult !== 'object' || Array.isArray(staticPathsResult)) {
throw new Error(
`Invalid value returned from getStaticPaths in ${page}. Received ${typeof staticPathsResult} ${expectedReturnVal}`
);
}
const invalidStaticPathKeys = Object.keys(staticPathsResult).filter(key => !(key === 'paths' || key === 'fallback'));
if (invalidStaticPathKeys.length > 0) {
throw new Error(
`Extra keys returned from getStaticPaths in ${page} (${invalidStaticPathKeys.join(', ')}) ${expectedReturnVal}`
);
}
if (!(typeof staticPathsResult.fallback === 'boolean' || staticPathsResult.fallback === 'blocking')) {
throw new Error(`The \`fallback\` key must be returned from getStaticPaths in ${page}.\n` + expectedReturnVal);
}
const toPrerender = staticPathsResult.paths;
if (!Array.isArray(toPrerender)) {
throw new Error(
`Invalid \`paths\` value returned from getStaticPaths in ${page}.\n` +
`\`paths\` must be an array of strings or objects of shape { params: [key: string]: string }`
);
}
在 buildStaticPaths 第一部分是获取 getStaticPaths 的返回值,并对其返回值进行检查:
getStaticPaths
可以为 async
方法 getStaticPaths
接受两个参数: locales
和 defaultLocale
{paths: Array, fallback: boolean | 'blocking'}
结构 而在拿到 toPrerender 之后, next.js 会将其转换为 prerenderPaths 和 encodedPrerenderPaths ,这两个 set 的数据集基本一致,只是一个 path 为已经被解码,一个没有,猜测是为了性能考虑空间换时间.
toPrerender.forEach(entry => {
if (typeof entry === 'string') {
entry = removeTrailingSlash(entry);
const localePathResult = normalizeLocalePath(entry, locales);
let cleanedEntry = entry;
if (localePathResult.detectedLocale) {
cleanedEntry = entry.slice(localePathResult.detectedLocale.length + 1);
} else if (defaultLocale) {
entry = `/${defaultLocale}${entry}`;
}
const result = _routeMatcher(cleanedEntry);
if (!result) {
throw new Error(`The provided path \`${cleanedEntry}\` does not match the page: \`${page}\`.`);
}
// If leveraging the string paths variant the entry should already be
// encoded so we decode the segments ensuring we only escape path
// delimiters
prerenderPaths.add(
entry
.split('/')
.map(segment => escapePathDelimiters(decodeURIComponent(segment), true))
.join('/')
);
encodedPrerenderPaths.add(entry);
} else {
// ...
}
});
针对 string 类型的 entry ,简单的处理下语言、路径即可.
const _validParamKeys = Object.keys(_routeMatcher(page));
if (typeof entry === 'string') {
// ...
} else {
const invalidKeys = Object.keys(entry).filter(key => key !== 'params' && key !== 'locale');
if (invalidKeys.length) {
throw new Error('...');
}
const { params = {} } = entry;
let builtPage = page;
let encodedBuiltPage = page;
_validParamKeys.forEach(validParamKey => {
const { repeat, optional } = _routeRegex.groups[validParamKey];
let paramValue = params[validParamKey];
if (
optional &&
params.hasOwnProperty(validParamKey) &&
(paramValue === null || paramValue === undefined || (paramValue as any) === false)
) {
paramValue = [];
}
if ((repeat && !Array.isArray(paramValue)) || (!repeat && typeof paramValue !== 'string')) {
throw new Error('...');
}
let replaced = `[${repeat ? '...' : ''}${validParamKey}]`;
if (optional) {
replaced = `[${replaced}]`;
}
builtPage = builtPage
.replace(
replaced,
repeat
? (paramValue as string[]).map(segment => escapePathDelimiters(segment, true)).join('/')
: escapePathDelimiters(paramValue as string, true)
)
.replace(/(?!^)\/$/, '');
encodedBuiltPage = encodedBuiltPage
.replace(
replaced,
repeat
? (paramValue as string[]).map(encodeURIComponent).join('/')
: encodeURIComponent(paramValue as string)
)
.replace(/(?!^)\/$/, '');
});
if (entry.locale && !locales?.includes(entry.locale)) {
throw new Error('...');
}
const curLocale = entry.locale || defaultLocale || '';
prerenderPaths.add(`${curLocale ? `/${curLocale}` : ''}${curLocale && builtPage === '/' ? '' : builtPage}`);
encodedPrerenderPaths.add(
`${curLocale ? `/${curLocale}` : ''}${curLocale && encodedBuiltPage === '/' ? '' : encodedBuiltPage}`
);
}
而对于 Object 类型的 entry ,则会先检查确保是 {params, locale} 结构,然后使用 params 对动态路由进行替换拼接。 _validParamKeys 是该动态路由页面中的参数的 key 数组。然后一样是路径和语言的处理。最终的返回值如下:
return {
paths: [...prerenderPaths],
fallback: staticPathsResult.fallback,
encodedPaths: [...encodedPrerenderPaths]
};
当需要时 next.js 就会使用这里的 paths 来生成对应的静态页面,从而实现动态路由的 SSG .
getStaticProps 、 getStaticPaths 相关的源码其实大部分都是在处理关于数据检查、处理这类的事情,因为这两个 API 的指责也都很简单: getStaticPaths 负责为动态路由的 SSG 场景提供页面列表, getStaticProps 则为 SSG 页面提供对应的页面数据.
最后此篇关于next.js源码解析-getStaticProps、getStaticPaths篇的文章就讲到这里了,如果你想了解更多关于next.js源码解析-getStaticProps、getStaticPaths篇的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
这是我的页面树 ├── _app.tsx ├── _document.tsx ├── index.tsx └── [type] └── [slug].tsx 这是 [slug] 页面的 get
我的“getStaticPaths”函数有问题。 当我尝试使用参数获取动态显示时,它显示为错误:在 getStaticPaths for/movies/[id] 中,没有以字符串形式提供必需的参数 (
使用 Next.js 导出静态页面,我在动态路由中得到它,如 pages/[id].js我在 getStaticPaths 中放入的任何路径部分将被创建。凉爽的。 列出每个页面是否更好: getSta
我正在处理以下代码: 这是一个 blogPost 动态页面:/post/[slug].tsx export const getStaticPaths: GetStaticPaths = async (
我对 Next.JS 的 getStaticPaths 有以下问题: 类型错误:cias.map 不是函数 /image/IVZDp.png 有人可以帮我解决这个问题吗? 代码: import Rea
我正在构建一个 Next.js 应用程序。这是我的文件结构: cpanearme -components -listitem.js -pages -home -inde
我正在寻找 docs有点暧昧。鉴于特许经营列表,我想在构建时呈现相关的特许经营详细信息页面,以避免在运行时访问 CMS/API,因为这些页面不会经常更改。 但是,似乎即使在构建时使用 getStati
我认为这是一个非常普通的问题,但我在谷歌上找不到任何东西。 我正在学习 NextJs(使用 TypeScript)并且我有一个站点成功地使用动态路由、SSR 和增量再生,所有设置和部署到 Vercel
下面是我的函数,其中我使用 getServerSideProps() 方法并根据我的 URL 中的参数 post_slug 动态获取数据。 // This gets called on every r
我正在构建小型电子商务网站,并设置了我的索引页面来生成静态 Prop 以及我的个人产品页面,因为我们还没有很多产品。我遇到的问题是每次用户点击时加载速度都很慢,好像我错过了 NextJS 中数据获取工
我收到此错误 "Error: getStaticPaths is required for dynamic SSG pages and is missing for 'xxx'"当我尝试在 NextJ
这个问题已经有答案了: Error: A required parameter (slug) was not provided as a string in getStaticPaths for /p
这个问题已经有答案了: Error: A required parameter (slug) was not provided as a string in getStaticPaths for /p
我已经设置了一个动态路由静态生成的页面组件(希望是这样吗?),它在开发模式下完美运行。我可以在我的 headless CMS (KeystoneJS) 中创建页面,并且可以在我的本地开发人员中创建和查
引用讨论 here .我遇到了类似的错误。当 fallback 设置为 false 时一切正常。但是,当fallback设置为true时,next js会抛出错误 throw new Error('
我是一名优秀的程序员,十分优秀!