惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

F
Full Disclosure
WordPress大学
WordPress大学
小众软件
小众软件
Cloudbric
Cloudbric
AWS News Blog
AWS News Blog
腾讯CDC
量子位
人人都是产品经理
人人都是产品经理
大猫的无限游戏
大猫的无限游戏
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
V
Vulnerabilities – Threatpost
Scott Helme
Scott Helme
Hugging Face - Blog
Hugging Face - Blog
博客园_首页
C
CXSECURITY Database RSS Feed - CXSecurity.com
The Hacker News
The Hacker News
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
IT之家
IT之家
Jina AI
Jina AI
Attack and Defense Labs
Attack and Defense Labs
S
SegmentFault 最新的问题
Simon Willison's Weblog
Simon Willison's Weblog
The Cloudflare Blog
阮一峰的网络日志
阮一峰的网络日志
T
Tailwind CSS Blog
Last Week in AI
Last Week in AI
博客园 - 【当耐特】
Google Online Security Blog
Google Online Security Blog
美团技术团队
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
V
Visual Studio Blog
罗磊的独立博客
L
LINUX DO - 最新话题
博客园 - Franky
博客园 - 叶小钗
Apple Machine Learning Research
Apple Machine Learning Research
The Last Watchdog
The Last Watchdog
J
Java Code Geeks
AI
AI
C
Cisco Blogs
酷 壳 – CoolShell
酷 壳 – CoolShell
C
Cyber Attacks, Cyber Crime and Cyber Security
Cisco Talos Blog
Cisco Talos Blog
博客园 - 三生石上(FineUI控件)
雷峰网
雷峰网
Help Net Security
Help Net Security
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
云风的 BLOG
云风的 BLOG
I
Intezer
S
Securelist

竹林里有冰的博客

小米 Xiaomi Book Pro 14 (Ultra X7) Linux 兼容性实测 | 竹林里有冰的博客 国内(大陆)版小米 FCM 熄屏断连:Rootless 环境下的尝试与可能的解决方案 | 竹林里有冰的博客 我没法访问 dl.google.com —— 记一次 TUN 下的网络 debug | 竹林里有冰的博客 Vercel 的缓存控制,你注意过吗? | 竹林里有冰的博客 小记 —— Caddy 在 Layer 4 上的流量代理实践 | 竹林里有冰的博客 你的域名后缀拖慢你的网站速度了嘛?——再谈 DNS 冷启动 | 竹林里有冰的博客 DNS 冷启动:小型站点的“西西弗斯之石” | 竹林里有冰的博客 HTTP/2 Server Push 已事实性“死亡”,我很怀念它 | 竹林里有冰的博客 Nuxt Content v3 中数组字段的筛选困境与性能优化 | 竹林里有冰的博客 后 OCSP 时代,浏览器如何应对证书吊销新挑战 | 竹林里有冰的博客 初试 Github Action Self-hosted Runner,想说爱你不容易 | 竹林里有冰的博客 DNS 解析延迟毁了我的图床优化 | 竹林里有冰的博客 Vue Markdown 渲染优化实战(下):告别 DOM 操作,拥抱 AST 与函数式渲染 | 竹林里有冰的博客 Vue Markdown 渲染优化实战(上):从暴力刷新、分块更新到 Morphdom 的华丽变身 | 竹林里有冰的博客 node-sass 迁移至 dart-sass 踩坑实录 | 竹林里有冰的博客 前端中的量子力学——一打开 F12 就消失的 Bug | 竹林里有冰的博客 2025 年,如何为 web 页面上展示的视频选择合适的压缩算法? | 竹林里有冰的博客 el-image 和 el-table 怎么就打架了?Stacking Context 是什么? | 竹林里有冰的博客 2025年,前端如何使用 JS 将文本复制到剪切板? | 竹林里有冰的博客 ssh 拯救世界——通过 ssh 隧道在内网服务器执行 APT 更新 | 竹林里有冰的博客 Cudy TR3000 吃鹅(daed)记 | 竹林里有冰的博客 使用 Cloudflare Workers 监控 Fedora Copr 构建状态 | 竹林里有冰的博客 基于 Cloudflare Workers 实现的在线服务状态检测告警系统 | 竹林里有冰的博客 构建部署在 Cloudflare Workers 上的 TG Bot | 竹林里有冰的博客 2024年,Firefox 是唯一还在坚持执行在线的 SSL 证书吊销状态检查的主流浏览器 | 竹林里有冰的博客 小爱课程表适配不完全指北——以 ZJUT 本科正方教务系统为例 | 竹林里有冰的博客 将博客从 waline v2 更新到 waline v3 | 竹林里有冰的博客 给家里云装上 Fedora 41 KDE 后,我是如何配置的 | 竹林里有冰的博客 为 Hexo 添加 follow 认证 | 竹林里有冰的博客 使用 GPT 对 waline 的评论进行审查 | 竹林里有冰的博客 基于 JavaScript 的 Hexo Fluid 主题 banner 随机背景图实现 | 竹林里有冰的博客 使用向日葵智能插座 C2 用电记录推算宿舍上次烧水时间 | 竹林里有冰的博客 使用 Caddy 反向代理 dockerhub 需要几步? | 竹林里有冰的博客 将 Rustdesk 中继服务从 Arch Linux 迁移至 Debian | 竹林里有冰的博客 自建图床小记五——费用 | 竹林里有冰的博客 自建图床小记四——上传脚本编写与图片迁移 | 竹林里有冰的博客 自建图床小记三—— SSL 证书的自动更新与部署 | 竹林里有冰的博客 自建图床小记二——使用 Workers 为 R2 构建 Restful API | 竹林里有冰的博客 自建图床小记一——图床架构与 DNS 解析 | 竹林里有冰的博客 在 Linux 下使用 mitmproxy 抓取安卓手机上的 HTTPS 流量 | 竹林里有冰的博客 为中柏 N100 小主机开启来电自启 | 竹林里有冰的博客 我的博客被完整地反向代理,并自动翻译成了繁体中文 | 竹林里有冰的博客 尝试体验 Fedora COPR 中的 allow SSH 功能 | 竹林里有冰的博客 在 Arch Linux 下配置使用 HP Laser 103w 打印机无线打印 | 竹林里有冰的博客 使用动态公网 ip + ddns 实现 rustdesk 的 ip 直连 | 竹林里有冰的博客 使用 Windows 虚拟机运行虚拟专用网客户端为 Linux 提供内网环境 | 竹林里有冰的博客 以 Archlinux 中 makepkg 的方式打开 rpmbuild | 竹林里有冰的博客 使用 Github Action 更新用于 rpm 打包的 spec 文件 | 竹林里有冰的博客 使用 Python 生成甘特图(Gantt Chart) | 竹林里有冰的博客 uniapp 中的图片预加载 | 竹林里有冰的博客 小记 - 尝试拼凑出 apt 仓库中的 deb 包下载地址 | 竹林里有冰的博客 在 Linux 下使用 mitmproxy 抓取 HTTPS 流量 | 竹林里有冰的博客 如何使用 docker 部署 onemanager | 竹林里有冰的博客 crontab 中简单的@语法糖 | 竹林里有冰的博客 备份 umami 数据库,并使用 TG Bot 保存 dump 文件 | 竹林里有冰的博客 在 JavaScript 中,箭头函数中的 this 指针到底指向哪里? | 竹林里有冰的博客 结合 Vue.js 与 php 完成的 web 期末大作业,讲讲前后端分离站点开发与部署中可能遇到的 CORS 跨域问题 | 竹林里有冰的博客 vuejs、php、caddy 与 docker —— web 期末大作业上云部署 | 竹林里有冰的博客 【翻译】使用 PHP 构建简单的 REST API | 竹林里有冰的博客 在 Hexo Fluid 主题中使用霞鹜文楷 | 竹林里有冰的博客 【翻译】GLWTPL——祝你好运开源许可证 | 竹林里有冰的博客 通过巴法云将向日葵智能插座接入米家,实现小爱同学远程控制 | 竹林里有冰的博客 使用 Root 后的安卓手机获取向日葵智能插座 C2 的开关 api | 竹林里有冰的博客 创建 b23.tv 追踪参数移除 bot | 竹林里有冰的博客 jinja2 中如何优雅地实现换行 | 竹林里有冰的博客 手动指定 python-selenium 的 driver path 以解决在中国大陆网络环境下启动卡住的问题 | 竹林里有冰的博客 从零开始的静态网页部署(到个人云服务器) | 竹林里有冰的博客 在运行OpenWRT的N1盒子上部署 QQBot | 竹林里有冰的博客 在浙工大宿舍使用路由器连接移动网络(校园网) | 竹林里有冰的博客 为红米 Redmi AC2100 路由器刷入 Padavan | 竹林里有冰的博客 Azure 教育订阅申请时遇到的麻烦 | 竹林里有冰的博客 执行 repo sync 后将 git-lfs 中的资源文件 checkout | 竹林里有冰的博客 隐式转发——骚套路建站方案 | 竹林里有冰的博客 在 vps 上配合 caddy 部署 siteproxy | 竹林里有冰的博客 onedrive(by abraunegg) —— 一个 Linux 下的开源 OneDrive 客户端(cli) | 竹林里有冰的博客 【翻译】关于2022年11月的事件的一些话[Z-Library] | 竹林里有冰的博客 【已过期】使用 vercel+supabase 免费部署 umami | 竹林里有冰的博客 我的博客部署方案 | 竹林里有冰的博客 使用 VirtScreen 将 Pad 作为副屏 | 竹林里有冰的博客 在 Archlinux 下使用 l2tp 协议连接校园网 | 竹林里有冰的博客 为 Element 添加自己喜欢的贴纸 | 竹林里有冰的博客 nodejs16:是我配不上 openssl 3 咯? | 竹林里有冰的博客 如何拯救失声的 hollywood | 竹林里有冰的博客 处理 fcitx5 的文字候选框在 tg 客户端上闪烁的问题 | 竹林里有冰的博客 使用caddy反向代理维基百科中文站点 | 竹林里有冰的博客 创建一个本地的 Fedora 镜像源 | 竹林里有冰的博客 好软推荐——FastOCR | 竹林里有冰的博客 抛弃PicGo,直接使用curl将图片上传到LskyPro | 竹林里有冰的博客 使用 Github Action 跑 rpmbuild | 竹林里有冰的博客 如何打出一个「-git」的rpm包 | 竹林里有冰的博客 雪藏在开源镜像站点中的那些常用却不为人知的软件 | 竹林里有冰的博客 在Fedora搭建jekyll环境——dnf module | 竹林里有冰的博客 pacman更新时遇到「GPGME 错误:无数据」 | 竹林里有冰的博客 Cutefish的前世今生 | 竹林里有冰的博客 wolai再打包遇到的问题--electron应用的dev判断机制 | 竹林里有冰的博客 Typora与我 | 竹林里有冰的博客 我是来吹CloudflareMirrors的 | 竹林里有冰的博客 deepin-elf-verify究竟是何物? | 竹林里有冰的博客 【翻译】请别再使用主题装饰我们的软件 | 竹林里有冰的博客 Waydroid on KDE 初体验 | 竹林里有冰的博客
Nuxt SSG 博客的尾斜杠到底怎么加? | 竹林里有冰的博客
竹林里有冰 · 2026-05-28 · via 竹林里有冰的博客

本站是用 Nuxt v4 + Nuxt Content v3 + i18n 搭出来的纯 SSG 博客。开站时随手定了一个看似无关紧要的策略——所有页面 URL 以 / 结尾

听起来一行配置就该完事的事,做下来才发现 Nuxt 在尾斜杠这件事上至今没有一个统一的官方开关nuxt/nuxt#15462 这个 issue 从 2022 年挂到现在),整套策略最后是靠六个不同层面拼出来的。这篇就把站点里所有跟 trailing slash 相关的配置完整盘一遍,留作给自己和后人的备忘。

为什么要尾斜杠#

简单提一句动机:

  • 同一篇文章 /2026/05/28/foo/2026/05/28/foo/ 在搜索引擎眼里理论上是两个 URL,要么你给一个 canonical,要么干脆只允许一种形态;
  • SSG 产物里"目录形态"更自然——about/index.htmlabout.html 更便于嵌套子页面、也更符合直觉;
  • 风格上,我个人更喜欢看 URL 末尾那个斜杠。
  • 在我的博客重构前使用的 hexo 框架就是这样的,我希望保留原有 URL 不变。

确定了"全部带斜杠"这个目标,下面要做的事就是让站点的每一个发出 URL 的地方、每一个接收 URL 的地方、每一个用 URL 做 key 的地方都遵守这条约定

Layer 1:SEO 层#

最先想到的是 SEO,所以 @nuxtjs/seo 的配置里:

// nuxt.config.ts
site: {
  trailingSlash: true,
}

这个开关只影响 canonical link、sitemap.xml、robots.txt、OpenGraph URL 等 SEO 模块生成的 URL。它不会改你页面里实际渲染出来的 <a href>,也不会拦截入站请求。但既然它是"对外发布我自己的 URL 形态",配上就对了。

Layer 2:出站链接#

第二层是页面里 <NuxtLink> 渲染出来的 href。Nuxt 4 提供了 experimental 配置:

// nuxt.config.ts
experimental: {
  defaults: {
    nuxtLink: { trailingSlash: 'append' }
  }
}

打开以后,全站任何 <NuxtLink to="/about"> 渲染出来都是 href="/about/"不管你 to 写没写斜杠

需要注意的边界:

  • 这个配置不影响 router.push('/about') 这种代码侧的导航,所以代码里手动 push 时还得自己拼。本站基本走 localePath() + NuxtLinkLocale,绕过了这个雷区。
  • localePath('/tags') 拼出来的 /tags,外面再手动 + '/' + tagName 的话,最后一段不带 / NuxtLink 的 'append' 会兜底再补一次。比如:
    <NuxtLink :to="`${localePath('/tags')}/${encodeURIComponent(tag)}`">
    

    实际渲染出来的 href 是 /tags/Foo/

Layer 3:硬编码的链接#

虽然有了 'append' 兜底,但项目里还是把所有硬编码的链接都直接写成带斜杠的形式,作为第二道防线:

<!-- FooterContent.vue -->
<NuxtLinkLocale to="/donate/" aria-label="Donate">
// Pagination.vue
function getPageUrl(page: number) {
  if (page < 1 || page > props.totalPages) return '#'
  return page === 1 ? `${props.urlPrefix}/` : `${props.urlPrefix}/page/${page}/`
}

养成这个习惯有个好处:将来如果 Nuxt 把 experimental.defaults.nuxtLink.trailingSlash 又改名了或者拿掉了(experimental API 嘛,懂的都懂),站点也不会因为这个一夜暴毙。

Layer 4:Prerender 产物落地形态#

SSG 阶段是 Nitro 在干活。它有个默认开启的配置叫 prerender.autoSubfolderIndex——会把 prerender 出来的每个页面落到 <path>/index.html,而不是 <path>.html

.output/public/
├── about/
│   ├── index.html
│   └── _payload.json
├── 2026/
│   └── 05/
│       └── 28/
│           └── nuxt-ssg-trailing-slash-hydration-trap/
│               ├── index.html
│               └── _payload.json
└── ...

这一步意味着,无论是 Vercel 这种 serverless 平台,还是 Nginx / Caddy,请求 /about/about/ 两种形态,静态文件服务器都能 fallback 到同一份 about/index.html——所以"用户输错斜杠也能开页"这件事根本不需要应用层兜底。

顺便:本站还显式列了几条 prerender route:

nitro: {
  prerender: {
    routes: [
      '/rss.xml',
      '/en/rss.xml',
      '/search/sections.json',
      '/tags/Vue.js',     // 带 . 的标签页,crawler 不会自动跟进
      ...Object.keys(blogConfig.redirects)
    ]
  }
}

这些是 crawler 抓不到、必须显式喂的,跟尾斜杠没直接关系,但放在这里作为完整的 nitro 配置一并列出。

Layer 5:入站 URL 规范化(纯前端)#

到这里 SEO、出站链接、产物落地都齐了,但有一类场景还没覆盖——用户手敲一个没斜杠的 URL(或者外部跳转过来),地址栏里挂着 /about,需要不需要把它改写成 /about/

经典做法是 HTTP 301。但本站是双平台部署(Vercel + Caddy),301 就得两份规则,能避免就避免。而且 Vercel 是把 SSG 产物放在 CDN 上的,301 写在 vercel.json 里也算半个绑定方案,不够纯粹。

所以这一层走纯前端:一个全局 client middleware。

// app/middleware/trailing-slash.global.ts
export default defineNuxtRouteMiddleware((to) => {
  if (import.meta.server) return
  if (to.path === '/' || to.path.endsWith('/')) return

  // 跳过 favicon.ico、rss.xml 这类带后缀的资源路径
  const lastSegment = to.path.slice(to.path.lastIndexOf('/') + 1)
  if (lastSegment.includes('.')) return

  return navigateTo(
    { path: to.path + '/', query: to.query, hash: to.hash },
    { replace: true }
  )
})

几个细节:

  • import.meta.server 直接 return。 SSG prerender 阶段 Nitro 自己已经归一化了;如果在 server 端再 navigateTo,可能在产物里写出非预期的 30x 跳转。
  • replace: true 让浏览器替换当前 history 条目,不会留一条"刚刚那个没斜杠的版本"的返回栈。
  • 排除带 . 的路径,避免误把静态资源也加上斜杠。

这层做完后,全链路 0 个 HTTP 301,配置上也不绑任何一家部署平台。

Layer 6:useAsyncData 的 key(隐藏的雷区)#

前五层做完,URL 的形态已经全部规范化,但还有一层非常隐蔽的地方需要照顾——useAsyncData 的 key

很容易写出这种代码:

const { data } = useAsyncData(
  `randomIndex${route.path}`,  // ← 雷
  async () => ...
)

问题是 route.path 在 SSR / client / prerender / SPA 导航这四种上下文里不一定一致。一旦 key 在 SSR 时算出 randomIndex/about、客户端水合时算出 randomIndex/about/,payload 命中失败,整个 useAsyncData 在客户端会重跑一遍,对应组件直接退化成 CSR。

本站的处理是:所有 useAsyncData 的 key 都不沾 route.path,要带路由信息就用 route.name + route.params

const route = useRoute()
const routeKey = `${String(route.name ?? 'unknown')}-${JSON.stringify(route.params)}`

const { data: randomIndex } = useAsyncData(
  `randomIndex-${routeKey}`,
  async () => Math.floor(Math.random() * appConfig.appearance.backgrounds.length)
)

route.name 是 vue-router 内部的路由名(i18n 自动生成的形如 about___zh),route.params 是动态段,两者在任何上下文都一致。最终构建出来的 _payload.json key 形如 randomIndex-about___zh-{},跟尾斜杠完全脱钩。

整体回顾#

整个站点的尾斜杠策略可以一句话总结:

Prerender 时让 Nitro 落到 xxx/index.html,SEO 由 site.trailingSlash 负责对外发布形态,出站链接由 nuxtLink.trailingSlash: 'append' 自动补斜杠(+ 硬编码做第二道防线),入站直链由全局 client middleware 兜底,useAsyncData 的 key 一律不依赖 route.path —— 全链路 0 个 HTTP 301

对照表:

配置/代码解决什么
SEOsite.trailingSlash: truecanonical / sitemap / OG URL
出站链接nuxtLink.trailingSlash: 'append'<NuxtLink> 渲染出来的 href
硬编码to="/donate/" 这种兜底 + 风格统一
产物落地nitro.prerender.autoSubfolderIndex(默认)/about/about/ 命中同一文件
入站 URLglobal client middleware用户输错 / 外链跳转的地址栏规范化
数据层useAsyncData 的 key 用 route.name + params避免两端 key 错位导致水合崩盘

小插曲:这套方案是怎么来的#

说起来这套配置并不是我开站时一次性想清楚的,最后两层(client middleware 和 useAsyncData 的 key)其实是前几天 debug 一个怪现象时被迫补上去的。

那天我打开自己博客的 /about/ 页,注意到一个怪事——背景图每次进来都"啪"地换一张。F12 一看,控制台挂着 Vue 的 Hydration completed but contains mismatches.

控制台报错控制台报错

明明 SSG 出来的纯静态产物,HTML 里 <div id="__nuxt"> 都齐齐整整,凭什么客户端不认账?

useAsyncData 是怎么命中 payload 的——key 一致就读 payload,不一致就重跑 fetch。那只能是 key 不一致。把构建产物的 _payload.json 抠出来看:

{"randomIndex/about": ...}

key 是 randomIndex/about没有尾斜杠。可这个文件本身在 .output/public/about/_payload.json,浏览器访问的 URL 是 /about/,客户端 route.path 拼出来的 key 是 randomIndex/about/——多了一个斜杠

Math.random() 在客户端重跑,DOM 与服务端渲染对不上,水合崩盘,整页 re-render。

罪魁祸首就是 route.path 在 SSR / client 两端因为 Nitro prerender 的归一化时机而不一致。修起来不难——useAsyncData 的 key 改用 route.name + params,跟路径解耦就完了。

修完才想起来,地址栏里那个没斜杠的 URL 还在挂着,于是又顺手把 client middleware 也加上,把"用户输错斜杠"这条路径也一并接住。

完了发现这是个值得正经写一篇下来留底的事——所以才有了这篇文章。