你以为Spring Boot统一异常处理能拦截所有的异常?

举报
码农小胖哥 发表于 2022/04/14 01:20:24 2022/04/14
【摘要】 通常我们在Spring Boot中设置的统一异常处理只能处理Controller抛出的异常。有些请求还没到Controller就出异常了,而这些异常不能被统一异常捕获,例如Servlet容器的某些异常。今天我在项目开发中就遇到了一个,这让我很不爽,因为它返回的错误信息格式不能统一处理,我决定找个方案解决这个问题。 Error...

通常我们在Spring Boot中设置的统一异常处理只能处理Controller抛出的异常。有些请求还没到Controller就出异常了,而这些异常不能被统一异常捕获,例如Servlet容器的某些异常。今天我在项目开发中就遇到了一个,这让我很不爽,因为它返回的错误信息格式不能统一处理,我决定找个方案解决这个问题。

ErrorPageFilter

Whitelabel Error Page

这类图相信大家没少见,Spring Boot 只要出错,体现在页面上的就是这个。如果你用Postman之类的测试出了异常则是:


   
  1. {
  2.   "timestamp""2021-04-29T22:45:33.231+0000",
  3.   "status"500,
  4.   "message""Internal Server Error",
  5.   "path""foo/bar"
  6. }

这个是怎么实现的呢?Spring Boot在启动时会注册一个ErrorPageFilter,当Servlet发生异常时,该过滤器就会拦截处理,将异常根据不同的策略进行处理:当异常已经在处理的话直接处理,否则转发给对应的错误页面。有兴趣的可以去看下源码,逻辑不复杂,这里就不贴了。

另外当一个 Servlet 抛出一个异常时,处理异常的Servlet可以从HttpServletRequest里面得到几个属性,如下:

异常属性

我们可以从上面的几个属性中获取异常的详细信息。

默认错误页面

通常Spring Boot出现异常默认会跳转到/error进行处理,而/error的相关逻辑则是由BasicErrorController实现的。


   
  1. @Controller
  2. @RequestMapping("${server.error.path:${error.path:/error}}")
  3. public class BasicErrorController extends AbstractErrorController {
  4.     //返回错误页面
  5.   @RequestMapping(produces = MediaType.TEXT_HTML_VALUE)
  6.  public ModelAndView errorHtml(HttpServletRequest request, HttpServletResponse response) {
  7.   HttpStatus status = getStatus(request);
  8.   Map<String, Object> model = Collections
  9.     .unmodifiableMap(getErrorAttributes(request, getErrorAttributeOptions(request, MediaType.TEXT_HTML)));
  10.   response.setStatus(status.value());
  11.   ModelAndView modelAndView = resolveErrorView(request, response, status, model);
  12.   return (modelAndView != null) ? modelAndView : new ModelAndView("error", model);
  13.  }
  14.     // 返回json
  15.  @RequestMapping
  16.  public ResponseEntity<Map<String, Object>> error(HttpServletRequest request) {
  17.   HttpStatus status = getStatus(request);
  18.   if (status == HttpStatus.NO_CONTENT) {
  19.    return new ResponseEntity<>(status);
  20.   }
  21.   Map<String, Object> body = getErrorAttributes(request, getErrorAttributeOptions(request, MediaType.ALL));
  22.   return new ResponseEntity<>(body, status);
  23.  }  
  24. // 其它省略
  25. }

而对应的配置:


   
  1. @Bean
  2. @ConditionalOnMissingBean(value = ErrorController.class, search = SearchStrategy.CURRENT)
  3. public BasicErrorController basicErrorController(ErrorAttributes errorAttributes,
  4.       ObjectProvider<ErrorViewResolver> errorViewResolvers) {
  5.    return new BasicErrorController(errorAttributes, this.serverProperties.getError(),
  6.          errorViewResolvers.orderedStream().collect(Collectors.toList()));
  7. }

所以我们只需要重新实现一个ErrorController并注入Spring IoC就可以替代默认的处理机制。而且我们可以很清晰的发现这个BasicErrorController不但是ErrorController的实现而且是一个控制器,如果我们让控制器的方法抛异常,肯定可以被自定义的统一异常处理。所以我对BasicErrorController进行了改造:


   
  1. @Controller
  2. @RequestMapping("${server.error.path:${error.path:/error}}")
  3. public class ExceptionController extends AbstractErrorController {
  4.     public ExceptionController(ErrorAttributes errorAttributes) {
  5.         super(errorAttributes);
  6.     }
  7.     @Override
  8.     @Deprecated
  9.     public String getErrorPath() {
  10.         return null;
  11.     }
  12.     @RequestMapping(produces = MediaType.TEXT_HTML_VALUE)
  13.     public ModelAndView errorHtml(HttpServletRequest request, HttpServletResponse response) {
  14.         throw new RuntimeException(getErrorMessage(request));
  15.     }
  16.     @RequestMapping
  17.     public ResponseEntity<Map<String, Object>> error(HttpServletRequest request) {
  18.         throw new RuntimeException(getErrorMessage(request));
  19.     }
  20.     private String getErrorMessage(HttpServletRequest request) {
  21.         Object code = request.getAttribute("javax.servlet.error.status_code");
  22.         Object exceptionType = request.getAttribute("javax.servlet.error.exception_type");
  23.         Object message = request.getAttribute("javax.servlet.error.message");
  24.         Object path = request.getAttribute("javax.servlet.error.request_uri");
  25.         Object exception = request.getAttribute("javax.servlet.error.exception");
  26.         return String.format("code: %s,exceptionType: %s,message: %s,path: %s,exception: %s",
  27.                 code, exceptionType, message, path, exception);
  28.     }
  29. }

直接抛异常,简单省力!凡是这里捕捉的到的异常大部分还没有经过Controller,我们通过ExceptionController中继也让这些异常被统一处理,保证整个应用的异常处理对外保持一个统一的门面。不知道你有没有更好的办法,欢迎留言讨论。

每天进步一点点

Oracle发布了Java SE支持路线图,Java 8 支持到2030年

2021-04-28

经过我翻来覆去的思想斗争了一个月,最后做出了一个明智的决定

2021-04-26

我踩过的Spring Boot统一返回体中的坑

2021-04-25

文章来源: felord.blog.csdn.net,作者:码农小胖哥,版权归原作者所有,如需转载,请联系作者。

原文链接:felord.blog.csdn.net/article/details/116311105

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。