解决Nginx配置React上线项目不能刷新问题

问题场景

基于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时,这个请求最终会到达nginxnginx匹配location时找不到/Home(也就是说nginx认为没有/Home这个资源,实际上也是如此),从而返回404页面。

  • 在 SPA 即单页面应用里,当用户与浏览器的后退按钮交互时,并没有真正导航到新的 HTML 页面。 而是在 window.onpopstate 上与 history.pushStatehistory.replaceState 等 API 进行交互。注意 window.onpopstate 只会在浏览器前进、后退时触发,点击刷新按钮完成的是一次导航过程,会按照路径发送真实的资源请求。现代浏览器在访问SPA应用时使用Back-Forward CacheHTML,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,cssjs会重新加载

解决方案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...

相关推荐
M ? A1 小时前
VuReact 编译器核心重构:统一管理组件元数据收集
前端·javascript·vue.js·react.js·重构·开源
M ? A1 小时前
Vue转React最佳工具对比:Vuera、Veaury与VuReact
前端·javascript·vue.js·经验分享·react.js
GISer_Jing2 小时前
告别手搓架构图!Excalidraw+AI Skills 高效绘制手绘风技术图
前端·人工智能·react.js
GISer_Jing2 小时前
React 18+ 高级特性实战与面试精讲
javascript·react.js·面试
彧翎Pro3 小时前
跨平台开发新选择:Flutter与React Native深度对比
flutter·react native·react.js
三声三视4 小时前
React 18 并发渲染实战:useTransition、Suspense 与自动批处理深度解析
前端·javascript·react.js
英俊潇洒美少年18 小时前
Vue、React.lazy、React 19 异步组件核心区别
javascript·vue.js·react.js
英俊潇洒美少年1 天前
React 实现 AI 流式打字机对话:SSE 分包粘包处理 + 并发优化
前端·javascript·react.js
绝世唐门三哥1 天前
React性能优化:memo、useMemo和useCallback全解析
前端·react.js·memo
F2E_Zhangmo1 天前
react native如何发送蓝牙命令
javascript·react native·react.js