SpringBoot Web开发中的异常处理自动配置原理解析

 更新时间:2025年10月27日 10:48:27   作者:zl979899  
本文给大家介绍SpringBoot Web开发中的异常处理自动配置原理解析,本文结合实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧

默认规则

1. 默认情况下,SpringBoot提供/error处理所有错误的映射

2. 对于机器客户端,它将生成JSON响应,其中包含错误,HTTP状态和异常消息的详细信息

  • 例如postman发送请求

3. 对于浏览器客户端,响应一个“whitelabel”错误视图,以HTML格式呈现相同的数据

4. 自定义错误视图解析规则

  • 完全替换默认行为,可以实现 ErrorController 并注册该类型的Bean定义,或添加ErrorAttributes类型的组件

定制错误处理逻辑

1. 自定义错误页

  • 替换默认的“whitelabel”错误视图,在templates在创建error文件夹,定义404和5xx的视图error/404.html error/5xx.html
  • 有精确的错误状态码页面就匹配精确,没有就找4xx.html或者N(错误状态码首数字)xx.html;如果都没有就触发白页

2. 方法一:@ExceptionHandler处理异常

  • 注解用来定义一个或多个具体的异常类型,发生异常时由此处理
  • @ControllerAdvice+@ExceptionHandler处理全局异常
  • 底层是 ExceptionHandlerExceptionResolver 支持的
/**
 * 处理整个web controller的异常
 */
@Slf4j
@ControllerAdvice
public class GlobalExceptionHandler {
    //定义可以处理的异常
    @ExceptionHandler({ArithmeticException.class,
            NullPointerException.class})
    public String handleArithException(Exception e){
        log.error("异常是:{}",e);
        //返回视图地址,也可以返回ModelAndView
        return "login";
    }
}

3. 方法二:@ResponseStatus+自定义异常

  • 指定发生异常时的响应状态码,自定义异常类型
  • 底层是 ResponseStatusExceptionResolver
  • 把ResponseStatus注解的信息组装成ModelAndView返回
    • 底层调用 response.sendError(statusCode, resolvedReason);
    • tomcat发送的/error
//异常响应状态码+异常原因
@ResponseStatus(value= HttpStatus.FORBIDDEN,reason = "用户数量太多")
public class UserTooManyException extends RuntimeException {
    public  UserTooManyException(){
    }
    public  UserTooManyException(String message){
        super(message);
    }
}

4.  Spring底层的异常

  • 如参数类型转换异常
  • DefaultHandlerExceptionResolver 处理框架底层的异常
    • 解析异常时做了一系列异常类型的判断
    • 每个异常处理都会执行response.sendError(.....),tomcat发送/error请求
    • 如果还是解析不了,就会响应tomcat的错误页面

5. 方法三:自定义实现 HandlerExceptionResolver 处理异常

  • @Order(value = Ordered.HIGHEST_PRECEDENCE) 将优先级设为最高
  • 如果不加优先级,默认是排列在自动配置的异常处理器之后,若默认的处理器可以处理异常,故自定义的异常处理器不会生效
  • 调用response.sendError(.....),tomcat发送/error请求
//优先级,数字越小优先级越高
@Order(value= Ordered.HIGHEST_PRECEDENCE)
@Component
public class CustomerHandlerExceptionResolver implements HandlerExceptionResolver {
    @Override
    public ModelAndView resolveException(HttpServletRequest request,
                                         HttpServletResponse response,
                                         Object handler, Exception ex) {
        try {
            //状态码+异常信息
            response.sendError(511,"我喜欢的错误");
        } catch (IOException e) {
            e.printStackTrace();
        }
        return new ModelAndView();
    }
}

6. 关于response.sendError()

  • 自己调用response.sendError(),请求会被转发给basicErrorController进行处理
  • 如果自己没有调用,并且异常没有任何人能够处理,tomact底层会自动调用response.sendError()将请求转发给basicErrorController处理。
  • 总之,三种方法最后都会调用sendError()转交给basicErrorController处理,ErrorViewResolver进行解析,最后会转到一个页面或者返回json数据。

异常处理自动配置原理

  • ErrorMvcAutoConfiguration 自动配置异常处理规则

1. 绑定配置文件的server spring.mvc 

2. 自动添加组件:类型:DefaultErrorAttributes -> id:errorAttributes

  • DefaultErrorAttributes:定义错误页面中可以包含哪些数据
  • DefaultErrorAttributes implements ErrorAttributes,HandlerExceptionResolver
  • 自定义错误页面属性

3. 自动添加组件:类型:BasicErrorController --> id:basicErrorController

  • 可以通过BasicErrorController 自定义页面跳转逻辑
  • 处理默认 /error 路径的请求;页面响应 new ModelAndView("error", model),
    • 要么响应一个HTML页面,要么响应一个json

  • 容器中有组件 View->id是error;(响应默认错误页)

  • 容器中放组件 BeanNameViewResolver(视图解析器);按照返回的视图名作为组件的id去容器中找View对象

4. 自动添加组件:类型:DefaultErrorViewResolver -> id:conventionErrorViewResolver

  • 如果发生错误,会以HTTP的状态码 作为视图名字(viewName),找到真正的页面
  • 比如error/404、5xx.html

异常处理步骤流程

  • 举例,执行一个controller方法,除数为0抛出异常

1. doDispatch()中执行controller目标方法

  • 目标方法运行期间有任何异常都会被catch、而且标志当前请求结束
  • 并且将异常赋值给 dispatchException
  • 即使目标方法发生异常,也执行视图解析流程

2. 处理handler发生的异常,处理完成返回ModelAndView

  •  遍历所有的 handlerExceptionResolvers,看谁能处理当前异常【HandlerExceptionResolver处理器异常解析器】
    • DefaultErrorAttributes先来处理异常,把异常信息保存到request域,并且返回null
    • 然后交由HandlerExceptionResolverComposite,遍历其中的异常处理器处理,但此例子默认无法处理
      • ExceptionHandlerExceptionResolver,使用注解@ExceptionHandler才能由此解析处理
      • ResponseStatusExceptionHandler,使用注解@ResponseStatus才能由此解析处理

3. 默认无法处理目标方法异常时,异常会被抛出

  • 底层就会发送 /error 请求,会被BasicErrorController处理

  • 解析错误视图;遍历所有的 ErrorViewResolver 看谁能解析

4. 模板引擎响应出错误视图

  • 默认的 DefaultErrorViewResolver ,作用是把响应状态码作为错误页的地址,error/500.html
  • 查找顺序:
    • '/templates/error/500.html' >
    • '/static/error/500.html' >
    • '/templates/error/5xx.html' >
    • '/static/error/5xx.html'

相关文章

  • SpringBoot+MySQL实现读写分离的多种具体方案

    SpringBoot+MySQL实现读写分离的多种具体方案

    在高并发和大数据量的场景下,数据库成为了系统的瓶颈。为了提高数据库的处理能力和性能,读写分离成为了一种常用的解决方案,本文将介绍在Spring Boot项目中实现MySQL数据库读写分离的多种具体方案,需要的朋友可以参考下
    2023-06-06
  • 关于jpa中无法删除onetomany中many问题的解决

    关于jpa中无法删除onetomany中many问题的解决

    这篇文章主要介绍了关于jpa中无法删除onetomany中many问题的解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-12-12
  • IntelliJ IDEA 2020.1.2激活工具下载及破解方法免费可用至2089年(强烈推荐)

    IntelliJ IDEA 2020.1.2激活工具下载及破解方法免费可用至2089年(强烈推荐)

    这篇文章主要介绍了IntelliJ IDEA 2020.1.2激活工具下载及破解方法免费可用至2089年(强烈推荐),本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-09-09
  • Mybatis中的PageHelper的执行流程分析

    Mybatis中的PageHelper的执行流程分析

    这篇文章主要介绍了Mybatis的PageHelper执行流程,本文给大家介绍介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-02-02
  • Java读取文件的简单实现方法

    Java读取文件的简单实现方法

    这篇文章主要介绍了Java读取文件的简单实现方法,通过一个读取txt格式的log文件为例,详细的讲述了Java读取文件的方法及原理,需要的朋友可以参考下
    2014-09-09
  • 使用MybatisPlus自定义模版中能获取到的信息

    使用MybatisPlus自定义模版中能获取到的信息

    这篇文章主要介绍了使用MybatisPlus自定义模版中能获取到的信息,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-05-05
  • java中@SuppressWarnings注解用法详解

    java中@SuppressWarnings注解用法详解

    这篇文章主要介绍了java中@SuppressWarnings注解用法详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-02-02
  • spring security环境搭建

    spring security环境搭建

    本文通过代码给大家介绍了spring security环境搭建的详细教程,非常不错,具有参考借鉴价值,需要的朋友参考下吧
    2017-09-09
  • 深入浅出的学习Java ThreadLocal

    深入浅出的学习Java ThreadLocal

    本文会基于实际场景介绍ThreadLocal如何使用以及内部实现机制。 具有很好的参考价值,下面跟着小编一起来看下吧
    2017-02-02
  • Mybatis Plus 增删改查的实现(小白教程)

    Mybatis Plus 增删改查的实现(小白教程)

    本文主要介绍了Mybatis Plus 增删改查,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-09-09

最新评论