问题场景
基于React
构建的单页应用中,我使用react-Router
中的BrowserRouter
完成页面跳转功能。打包上传至服务器,使用nginx
配置时跳转路由无法刷新。例如,https://example.com
作为应用入口的话,跳转至 https://example.com/Home
后无法刷新,报错为nginx 404
界面。 打包文件在/home/front/upload
下,
nginx
配置如下:
原因
React-Router
实现BrowserRouter
路由跳转基于history
,假设当我们点击导航栏上的按钮,初始页面 https://example.com
跳转至 https://example.com/Home
。这个过程是由浏览器来完成的。但是,点击刷新按钮,浏览器会进行一次导航,访问https://example.com/Home
时,这个请求最终会到达nginx
,nginx
匹配location
时找不到/Home
(也就是说nginx
认为没有/Home
这个资源,实际上也是如此),从而返回404页面。
- 在 SPA 即单页面应用里,当用户与浏览器的后退按钮交互时,并没有真正导航到新的 HTML 页面。 而是在
window.onpopstate
上与history.pushState
和history.replaceState
等 API 进行交互。注意window.onpopstate
只会在浏览器前进、后退时触发,点击刷新按钮完成的是一次导航过程,会按照路径发送真实的资源请求。现代浏览器在访问SPA应用时使用Back-Forward Cache
对HTML,Css,JS
等资源暂存起来。用户进行页面跳转的时候,浏览器可以不必再次解析HTML、Css、JS
等资源。前端只需要Ajax获取数据,使用连接抓取对象存储服务上的资源即可。
解决方案1:nginx配置try_files规则
增加如下的try_files
规则
nginx
官网对try_files
的解释如下:docs.nginx.com/nginx/admin...
-
try_files
try_files 指令可用于检查指定的文件或目录是否存在;如果存在,NGINX 会进行内部重定向;如果不存在,则返回指定的状态代码。 -
文件以 URI 的形式指定,并使用在当前位置或虚拟服务器上下文中设置的根指令或别名指令进行处理。在这种情况下,如果与原始 URI 相对应的文件不存在,NGINX 会内部重定向到最后一个参数指定的 URI。
这样配置之后,我们的刷新请求会被内部重定向到react
应用的/Home
路由下,最后nginx
转发这次请求的回复报文给用户。 这样刷新后,html,css
和js
会重新加载
解决方案2:使用哈希路由
哈希路由,形如https://example.com/path/#/aaa/bbb
,将一部分路径使用#
拼接在真实路由的后面,当井号 #
后面的路径发生变化时,浏览器并不会重新发起请求,而是会触发 onhashchange
事件。
实际上哈希路由最开始是作为锚点
应用在页面中的,目的是对文档、协议书等长页面的词条进行定位和跳转的机制,后来被用作SPA应用,通过onhashchange
事件,判断当前url下哪一个组件应该被渲染,从而实现页面的跳转,实际上是一种页面组件的切换。
浏览器使用哈希路由发起导航时,hash模式下仅hash符号之前的内容会被包含在http请求中,如https://example.com/path/#/aaa/bbb
,那么浏览器会对https://example.com/path/
发起请求,得到页面后,再根据哈希路由完成定位和页面展示。这样在之前的例子中,我们将Home页面放到 https://example.com/#/Home
下,刷新页面时,nginx
收到的请求为https://example.com/
,能够通过配置的location返回页面到前端,后续的页面展示由前端完成,因此解决了返回404错误的问题。
不只是React-Router,对Vue-Router也是同理。
链接
react.js application showing 404 not found in nginx server
stackoverflow.com/questions/4...
Trying Several Options
docs.nginx.com/nginx/admin...
浅谈前端路由原理hash和history
juejin.cn/post/699384...
Vue/React 项目部署到服务器后,刷新页面出现404报错
blog.csdn.net/weixin_7331...