原文地址:https://www.jianshu.com/p/b06584591086

0、简述

在学习这个学习笔记之前最好能够对spring mvc以及Tomcat有些了解,这样理解起来更加方便,如果需要知道最直接的解决方案,拖到最底部看样例代码即可。

介绍了springboot的白页出现的真正原因,主要是没有合适的匹配情况出现404情况,然后跳转到系统默认的第一个ErrorPage,也就是白页内容上,然后根据其特定分别从三个角度,1、拦截器,2、新ErrorPage,3、自定义/error路由 去解决该问题,并且介绍各自方法的优缺点,其中还有介绍到循环页面错误的本质原因等情况

1、Whitelabel Error Page 白页

什么叫Whitelabel Error Page(也叫白页),就是SpringBoot中HTTP请求出现异常的说明页,如下图

image

白页内容会展示状态码、path、以及错误原因等情况,但是真正发布在线上生成环境一般不允许出现这样的情况,更多的是自定义的404页面或者500页面等。

那么现在我们就来了解下什么情况会产生白页的情况,以及如何解决这种情况。我们就以404的情况去了解其原因。

直接来到DispatcherServlet类的protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception方法,其中包含的代码片段

mappedHandler = getHandler(processedRequest);
// 找到合适的请求处理器
if (mappedHandler == null || mappedHandler.getHandler() == null) { // 原则上如果没有找到则会进入到这里,并且设置response的状态码为404 // 但是经过调试并没有进入到这里 noHandlerFound(processedRequest, response); return; }

在getHandler方法中会遍历当前web容器中的HandlerMapping,找出合适的处理器Handler

image

image

由上图可以很明显的知道便利出的当前Handler是SimpleUrlHandlerMapping,因为其中的url中包含了/**,所有的url都可以被匹配出,不会进入到后面的noHandlerFound中,适配处理器HandlerAdapter是HttpRequestHandlerAdapter实例化的对象

在mv = ha.handle(processedRequest, response, mappedHandler.getHandler())中没法找到对应的resource,设置response的状态码为404,具体可看ResourceHttpRequestHandler类的handleRequest方法

现在就相当于该请求设置了状态码为404,其他并没有做什么,mv也是为null的

这时候需要回到Tomcat的调用流程内,如果对Tomcat的调用流程请求的同学应该知道,Tomcat在连接器收到Socket套接字请求包装成为request、response等信息交由Engine->Host 等组件一层一层传递,再由各个组件的Pipeline管道接收,后续各自的Valve(阀门)一层一层的过滤处理。

这个时候来到StandardHostValve类的private void status(Request request, Response response)方法

private void status(Request request, Response response) { int statusCode = response.getStatus(); // 查看当前状态码,当前样例是404 // 获取当前上下文 Context context = request.getContext(); if (context == null) { return; } if (!response.isError()) { // 当前请求没错误 // 是一个原子类AtomicInteger errorState,如果大于0则认为遇到错误 return; } ErrorPage errorPage = context.findErrorPage(statusCode); // 这个地方以后再说,就是解决方案的一种,设置错误页 if (errorPage == null) { // Look for a default error page errorPage = context.findErrorPage(0); } if (errorPage != null && response.isErrorReportRequired()) { ...

image

结合代码和图,再看白页清楚的写着This application has no explicit mapping for /error,路由为\error,这个缘由就是来自这里,然后进行forward跳转,路由地址是\error

image

后面使用了SpringBoot提供的白页mv,渲染生产我们所看到的白页页面内容

至此,整个的流程也已经执行完成,总结一下就是请求一个不存在的链接,被发现是404请求之后转发到/error的请求上

那么解决方案就很简单了,有三种方案,不过这三种方案均是从不同的角度去解决该问题

  • 添加拦截器
  • 添加ErrorPage
  • 添加/error 路径

2、解决白页问题

2.1、添加拦截器

public class CustomHandlerInterceptor implements HandlerInterceptor { @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { // 一定得为true,否则拦截器就无法生效了 // 当然可以随意各种对url的拦截处理 return true; } @Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { if (modelAndView != null) { // 防止出现空指针 // 在springboot中如果是错误页肯定不会出现mv为null的情况 modelAndView.setViewName("/err"); // 注意:该请求只是测试试用,并没有实际的意义 } } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { } } @Bean public WebMvcConfigurerAdapter customMvcConfigurerAdapter (){ return new WebMvcConfigurerAdapter() { @Override public void addInterceptors(InterceptorRegistry registry) { registry.addInterceptor(new CustomHandlerInterceptor()).addPathPatterns("/**"); // 添加拦截器 super.addInterceptors(registry); } }; }

拦截器在捕获到/error 的请求之后,强制修改mv,使得最后渲染试用的mv是我们自定义设置的,而不是白页内容,其中白页本身的mv会经过ContentNegotiating视图解析器处理成为ErrorMvcAutoConfiguration$WhitelabelErrorViewConfiguration

注意这整个过期其实经过3次HTTP请求处理的,如下图是使用HTTP事件监听打印出的日志信息

image

经历了/abc==>跳转/err==>跳转/error(并不显示该内容,因为发送到浏览器的内容已经经过/err 渲染完成

其中真正的调用处理流程是/abc 没有发现合适的handler,后选择交由/error 路径处理,只是后面又被拦截器拦截处理,转发给了/err 处理

缺点:会拦截该路由的所有请求,包含静态资源文件,对纯提供接口的后端服务没太多影响,其他的服务会有影响的

2.2、添加ErrorPage

添加合适的ErrorPage就不会出现跳转到拦截器默认的/error 路径,而是跳转到自定义的ErrorPage上,这点在上面的status方法已经介绍其原因了

@Bean
public EmbeddedServletContainerCustomizer containerCustomizer() { return new EmbeddedServletContainerCustomizer() { @Override public void customize(ConfigurableEmbeddedServletContainer configurableEmbeddedServletContainer) { ErrorPage errorPage400 = new ErrorPage(HttpStatus.BAD_REQUEST,"/400"); ErrorPage errorPage404 = new ErrorPage(HttpStatus.NOT_FOUND,"/404"); ErrorPage errorPage500 = new ErrorPage(HttpStatus.INTERNAL_SERVER_ERROR,"/500"); configurableEmbeddedServletContainer.addErrorPages(errorPage400,errorPage404,errorPage500); } }; } // ======= 分割线 @ApiOperation("404请求") @GetMapping("404") public String e404() { return "404"; }

上述代码添加了几个错误页ErrorPage跳转的路径以及其对应的HTTP错误码,我们当前的样例肯定是跳转到/404连接上去了,再执行怎么又报错了,原则上来说应该显示404.html的内容文件,同时显示的是经典的Tomcat错误页了,如下图页面展示以及日志输出的内容

image

image

这个就是循环跳转的问题

当系统没有指定明确的视图解析器之后,系统便会使用自带的默认解析器InternalResourceView,在渲染前会去校验当前url的问题,如果发现请求的url和目的url是一致的情况,就会认定转发自身,出现Circular view path的问题

protected String prepareForRendering(HttpServletRequest request, HttpServletResponse response) throws Exception { String path = getUrl(); if (this.preventDispatchLoop) { String uri = request.getRequestURI(); if (path.startsWith("/") ? uri.equals(path) : uri.equals(StringUtils.applyRelativePath(uri, path))) { throw new ServletException("Circular view path [" + path + "]: would dispatch back " + "to the current handler URL [" + uri + "] again. Check your ViewResolver setup! " + "(Hint: This may be the result of an unspecified view, due to default view name generation.)"); } } return path; }

那么如何解决呢,肯定得从根本目的出发

  • 添加模板解析器,这样就不会使用默认解析器了
  • 修改跳转路径

具体的解决方案就自行了解,本文不使用模板引擎渲染,直接展示body数据

@RequestMapping("/")
@RestController
public class ErrorController { @ApiOperation("404请求") @GetMapping("404") public String e404() { System.out.println("404............"); return "这真的是一个404页面,你看看"; }

image

2.3、添加/error 路径

上面已经知道了,既然默认的系统跳页至/error,并且会完成数据渲染,那么我们自定义个/error的路由不就可以了,而且避免了静态资源找不到的问题,不过注意这里存在一个问题,具体看下图

image

image

先自行添加定义了一个很简单的/error路径的处理方法,但是在springboot启动时,共有3个/error路径的处理器方法,而且同时隶属于同一个handle中,恰好经过URL路由规则处理,选择了autoconfigure里面的handle,这肯定导致了我们自定义的/error 无效

image

经过测试也确实无效,依旧显示白页,那么如何解决呢?同样的方法有多种

  • 根据路由匹配规则,修改适当内容,使得最后选择处理器的时候达到我们自定义的处理器上,不过这点难度很大吧,需要对spring mvc 本身的路由映射规则了解的相当清楚,确保url映射处理的优先级问题等
  • 上述我们已经知道了这三个/error是在RequestMappingHandlerMapping路由映射器中,我们可以使得自定义的处理器不存放在该路由映射其中,并且使得spring在轮询时,优先处理约定的路由映射器即可,可是事实上handlerMapping中的BeanNameUrlMapping还在RequestMappingHandlerMapping之后,如果再去修改这个顺序,同样难度很大

    image

EndpointHandlerMapping 是springboot的actuator模块中的端点,自定义端点难度较大而且不适用在当前项目中

使用SimpleUrlHandlerMapping针对在springboot中不太合适,如果使用xml配置可直接设置其url会很方便的,如果强加在springboot中使用注解的方式需要额外配置,如下代码

@RequestMapping("/")
@Controller
public class SimpleUrlController { private static final String ERROR_PATH = "/error"; @Resource private SimpleUrlHandlerMapping simpleUrlHandlerMapping; @PostConstruct public void init() { Map<String, Object> map = new HashMap<>(); map.put(ERROR_PATH, this); simpleUrlHandlerMapping.setUrlMap(map); simpleUrlHandlerMapping.initApplicationContext(); // 重新调用simpleurl映射 } @GetMapping("/error") @ResponseBody public String error(HttpServletRequest request) { System.out.println("SimpleUrlController"); Enumeration<String> attributes = request.getAttributeNames(); Map<String, Object> map = new HashMap<String, Object>(); while (attributes.hasMoreElements()) { String name = attributes.nextElement(); if (name.startsWith("java")) { // spring本身的属性不宜对外暴露,切记! Object value = request.getAttribute(name); map.put(name, value); } } return JSON.toJSONString(map); } }

image

虽然完成了/error 注入到SimpleUrlHandlerMapping中,可是即使添加了额外的配置依旧会出现无适配器的错误这种方法并不适用

回过头通过对BasicErrorController的观察,我们可以自行继承ErrorController接口完成

@RequestMapping("")
@Controller
public class CustomErrorController implements ErrorController { private static final String ERROR_PATH = "/error"; @GetMapping(ERROR_PATH) @ResponseBody public String error(HttpServletRequest request) { System.out.println("CustomErrorController"); Enumeration<String> attributes = request.getAttributeNames(); Map<String, Object> map = new HashMap<String, Object>(); while (attributes.hasMoreElements()) { String name = attributes.nextElement(); if (name.startsWith("java")) { // spring本身的属性不宜对外暴露,切记! Object value = request.getAttribute(name); map.put(name, value); } } return JSON.toJSONString(map); } @Override public String getErrorPath() { return ERROR_PATH; } }

image

到这里整个的过程就算是完成了

转载于:https://www.cnblogs.com/davidwang456/articles/11011461.html

SpringBoot Whitelabel Error Page的根本原因,三种解决方案以及其特点相关推荐

  1. SpringBoot Whitelabel Error Page 错误

    SpringBoot Whitelabel Error Page 错误 错误描述 解决方案 Controller类中添加注解@Controller

  2. spring boot: Whitelabel Error Page(小白的终极解决方案)

    假如你刚刚入手springboot,而且前面的spring.spring mvc都开始有点忘记了 此时你遇到了这个问题 Whitelabel Error Page 当你绝望无助的搜索了全网的解决方案的 ...

  3. springboot 连接不上 redis 的三种解决方案!

    针对于springboot 连接不上 redis 这种情况,首先,我们最简单直接的方法就是需要确认Redis是否已经正常启动(验证方法:如果安装在Linux下的话可以使用ps-ef|grep redi ...

  4. SpringBoot:Whitelabel Error Page 404

    项目:SpringBoot + VUE 原因: 线上环境发版之后就进不去系统了,一直报如上错误,重新打包编译也没用,原来是因为前端下载了新的依赖没有上传更新后的版本号,导致如上错误发生. 解决办法: ...

  5. SpringBoot:如何处理SprintBoot提示Whitelabel Error Page以及了解原因?

    目录 QUESTION:如何处理SprintBoot提示Whitelabel Error Page? ANSWER: 一:产生原因 二:如何处理 2.1Whitelabel Error Page 白页 ...

  6. springboot启动报Whitelabel Error Page

    Whitelabel Error Page This application has no explicit mapping for /error, so you are seeing this as ...

  7. SpringBoot报错Unsatisfied dependency expressed through field userMapper和Whitelabel Error Page解决方案

    搜索下面的错误信息得到解决方法 Unsatisfied dependency expressed through field 'userMapper' 项目结构如下 解决办法 在启动类中加入注解 里面 ...

  8. SpringBoot页面出现 Whitelabel Error Page

    我们运行SpringBoot项目之后需要通过Tomcat进行访问,但是我们访问的时候出现了Whitelabel Error Page的错误,我们该如何解决呢??? 错误页面 究其原因是我们的主程序缺少 ...

  9. springboot访问页面显示Whitelabel Error Page

    背景描述 问题 项目 代码 配置文件 访问登录页面 解决 总结 背景描述 问题 新建springboot(2.5.7版本)项目配置好mvc,访问页面显示Whitelabel Error Page 项目 ...

最新文章

  1. Spring Cloud Greenwich版本已发布!
  2. DIY穷人版谷歌眼镜,自定义手势操控,树莓派再一次被开发新玩法
  3. 基于锁相环的定时误差调整
  4. 2021-5-18大搜车
  5. leetcode 121
  6. 数字图像处理课设图像的锐化_数字图像处理图像锐化处理.ppt
  7. java 一元稀疏多项式简单计算器_一元稀疏多项式简单的计算器
  8. 面试稳了!BATJ 等大厂 400+ 道面试题全汇总!
  9. JavaScriptjQuery.HTML5事件
  10. DB2数据库迁移,数据库导入导出
  11. 第十三届“恩智浦”杯全国大学生智能汽车竞赛-信标对抗组比赛总结
  12. 反射修改jsessionid实现Session共享
  13. 马哥教育SRE笔记【作业】week04
  14. web 前端签名插件_手写签名插件—jSignature
  15. TFN全新推出的全功能 手持式频谱分析仪 RMT系列 不仅可干扰定位 还可路测
  16. 如何去除图片雾化?给你推荐图片去雾怎么去除的方法
  17. vue下利用canvas实现在线图片标注
  18. 腾讯云服务器网站搭建教程(太简单了)
  19. 电视机尺寸与观看距离
  20. 带宽、速率(波特率、比特率)和码元宽度简述

热门文章

  1. 最长公共子序列LCS[C++题解]
  2. 安卓实训项目:基于储存卡音乐播放器实训报告5.0
  3. oracle分区list,Oracle 分区表中存在range-list表分区时遇到问题及解决办法
  4. uos系统断网怎么安装mysql_【学习笔记】 UOS安装MySQL
  5. python列表实现原理_Python动态类型实现原理及过程解析
  6. php-dev离线安装,局域网 pm2 离线安装
  7. yolov3 anchor 理解
  8. java todo error_java基础-异常
  9. 子数组的最大累加和问题
  10. 数据挖掘流程(六):写报告