手机端刷新总是 404?你需要知道 SPA Fallback 规则

记录一次手机端下拉刷新页面返回 404 的排查过程,以及 Vue 单页应用在 Vercel 上的正确部署姿势。

问题

论坛上线后,在电脑上浏览一切正常------点击链接跳转、前进后退都没问题。但用手机打开,上滑刷新页面时,浏览器直接显示 404 Not Found。

奇怪的是,同一个页面在电脑上刷新是正常的,为什么手机端不行?

排查

手机端上滑刷新,本质上是浏览器重新向服务器请求当前 URL。比如用户在 /post/123 这个页面刷新,浏览器就会向 Vercel 服务器请求 /post/123 这个文件。

问题来了------Vercel 服务器上根本没有 /post/123 这个文件。Vue 是单页应用,所有路由都是前端用 Vue Router 模拟出来的,服务器上只有一个 index.html。所以 Vercel 收到这个请求后,直接返回了 404。

那为什么电脑端点击链接跳转没问题?因为点击链接是 Vue Router 在浏览器端做的"假"页面切换,根本没有向服务器发请求。所以不会触发 404。但刷新页面是浏览器行为,它会绕过前端路由,直接向服务器要文件。

原理

单页应用的路由分两种:

路由方式 谁处理的 会发请求吗
点击链接跳转 Vue Router(前端) 不会
刷新页面 / 直接访问 URL 服务器

服务器收到请求后,发现没有对应的文件,就返回 404。这就是根因。

解决

在项目根目录的 vercel.json 里,加一条 Rewrite 规则。这个文件之前已经有一条规则用来转发 API 请求,现在再加一条处理页面路由:

json 复制代码
{
  "rewrites": [
    {
      "source": "/api/:path*",
      "destination": "https://xxx.onrender.com/api/:path*"
    },
    {
      "source": "/((?!api).*)",
      "destination": "/index.html"
    }
  ]
}

第二条规则的含义是 :所有不以 /api 开头的请求,全部返回 index.html

这样当用户在 /post/123 刷新时,Vercel 不再返回 404,而是把 index.html 交给浏览器。浏览器拿到之后加载 Vue 和 Vue Router,Router 看到地址栏是 /post/123,就渲染出对应的帖子详情页。

总结

这个问题本身不复杂,但暴露了一个很容易被忽略的部署细节:单页应用必须配置 SPA Fallback 规则 。不管是 Vercel、Netlify 还是 Nginx,都需要告诉服务器:找不到文件时,把 index.html 交给前端,让前端路由接管。

以后再搭新项目,这个规则应该在第一次部署时就配好,而不是等用户反馈 404 了再去修。不同平台的配置方式不同,但核心思路都一样:

平台 配置方式
Vercel vercel.json 里加 Rewrite 规则
Netlify 项目根目录加 _redirects 文件:/* /index.html 200
Nginx try_files $uri $uri/ /index.html;

这就是这次踩坑学到的东西------不是修了一个 Bug,而是补齐了一个部署常识。

相关推荐
竹林8188 分钟前
用 The Graph 查询链上数据实战:从手搓 RPC 到 Subgraph,我的 NFT 项目数据加载快了 10 倍
前端·javascript
妙码生花25 分钟前
从 PHP 到 AI + Golang,程序员自救转型手记(十九):点选验证码代码逐行目检
前端·后端·go
Awu12271 小时前
⚡从零开发 Agent CLI(五)实现一个可治理、可扩展的工具系统
前端·人工智能·claude
咪库咪库咪2 小时前
Vue3-生命周期
前端
莪_幻尘2 小时前
你的 AI Skill 越多越蠢?Token 上下文爆炸的求生指南
前端·ai编程
lichenyang4532 小时前
从 has.echo 到异步 API 注册表:一次 ASCF API 回调不触发的排查复盘
前端
林瞅瞅3 小时前
Nuxt3 项目部署 Nginx 防盗链后特定 JS 文件 403 问题修复方案
前端
kyriewen3 小时前
别再每次都 Google 了:我整理了前端日常最常踩的 10 个 Git 坑,附速查表
前端·javascript·git
一颗奇趣蛋3 小时前
Web 视频开发完全指南:从入门到精通
前端
非洲农业不发达4 小时前
windows终端体验大升级,让你拥有macos级别的美化
前端·后端