详解SpringBoot异常处理流程及原理


Posted in Java/Android onJune 21, 2021

异常处理流程

执行目标方法,目标方法运行期间有任何异常都会被catch捕获,并标志当前请求结束,dispatchException抛出异常

详解SpringBoot异常处理流程及原理

进入视图解析流程,并渲染页面,发生异常时,参数mv为空,传入捕获的异常dispatchException

详解SpringBoot异常处理流程及原理

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

详解SpringBoot异常处理流程及原理

(1)遍历所有的HandlerExceptionResolvers,找到可以处理当前异常的解析器来解析异常

详解SpringBoot异常处理流程及原理

(2)调用resolveException解析异常,传入requestresponse对象,哪个方法,发生的异常,然后自定义异常处理返回ModelAndView

详解SpringBoot异常处理流程及原理

(3)系统默认的异常解析器

详解SpringBoot异常处理流程及原理

DefaultErrorAttributes先来处理异常,把异常信息保存到request域并返回null

详解SpringBoot异常处理流程及原理

ExceptionHandlerExceptionResolver用来处理标注了@ExceptionHandler注解的方法异常

ResponseStatusExceptionResolver用来处理标注了@ResponseStatus注解的方法异常

DefaultHandlerExceptionResolver默认的处理器异常解析器,处理一些常见的异常

(4)如果没有任何解析器能够处理异常,异常就会抛出

详解SpringBoot异常处理流程及原理

(5)如果没有任何解析器能够处理当前异常,最终就会发送/error请求,将保存的异常信息转发到/errorBasicErrorController专门来处理/error请求,BasicErrorController会遍历所有的ErrorViewResolver解析错误视图,如果没有自定义的错误视图解析器,就会使用默认的DefaultErrorViewResolver,会把响应码作为错误页的地址,模板引擎最终响应这个页面。

几种异常处理方式及原理

1.自定义错误页,error/404.htmlerror/5xx.html。有精确的错误状态码页面就匹配精确,没有就找 4xx.html,如果都没有就触发白页

2.使用@ControllerAdvice@ExceptionHandler处理全局异常,底层是ExceptionHandlerExceptionResolver 支持的

详解SpringBoot异常处理流程及原理

3.使用@ResponseStatus和自定义异常。底层是 ResponseStatusExceptionResolver ,底层调用 response.sendError(statusCode, resolvedReason),Tomcat会收到一个error。请求最后new一个空的ModelAndView返回,这样任何处理解析器都处理不了当前的异常,最终就会发送/error请求,BasicErrorController专门来处理/error请求,适配4xx.html或者5xx.html页面

详解SpringBoot异常处理流程及原理
详解SpringBoot异常处理流程及原理
详解SpringBoot异常处理流程及原理

4.Spring底层的异常,如参数类型转换异常。底层是DefaultHandlerExceptionResolver 处理框架底层的异常,底层也是response.sendError(HttpServletResponse.SC_NOT_ACCEPTABLE),Tomcat会收到一个error。请求最后new一个空的ModelAndView返回,这样任何处理解析器都处理不了当前的异常,最终就会发送/error请求,BasicErrorController专门来处理/error请求,适配4xx.html或者5xx.html页面

protected ModelAndView doResolveException(
			HttpServletRequest request, HttpServletResponse response, @Nullable Object handler, Exception ex) {

		try {
			if (ex instanceof HttpRequestMethodNotSupportedException) {
				return handleHttpRequestMethodNotSupported(
						(HttpRequestMethodNotSupportedException) ex, request, response, handler);
			}
			else if (ex instanceof HttpMediaTypeNotSupportedException) {
				return handleHttpMediaTypeNotSupported(
						(HttpMediaTypeNotSupportedException) ex, request, response, handler);
			}
			else if (ex instanceof HttpMediaTypeNotAcceptableException) {
				return handleHttpMediaTypeNotAcceptable(
						(HttpMediaTypeNotAcceptableException) ex, request, response, handler);
			}
			else if (ex instanceof MissingPathVariableException) {
				return handleMissingPathVariable(
						(MissingPathVariableException) ex, request, response, handler);
			}
			else if (ex instanceof MissingServletRequestParameterException) {
				return handleMissingServletRequestParameter(
						(MissingServletRequestParameterException) ex, request, response, handler);
			}
			else if (ex instanceof ServletRequestBindingException) {
				return handleServletRequestBindingException(
						(ServletRequestBindingException) ex, request, response, handler);
			}
			else if (ex instanceof ConversionNotSupportedException) {
				return handleConversionNotSupported(
						(ConversionNotSupportedException) ex, request, response, handler);
			}
			else if (ex instanceof TypeMismatchException) {
				return handleTypeMismatch(
						(TypeMismatchException) ex, request, response, handler);
			}
			else if (ex instanceof HttpMessageNotReadableException) {
				return handleHttpMessageNotReadable(
						(HttpMessageNotReadableException) ex, request, response, handler);
			}
			else if (ex instanceof HttpMessageNotWritableException) {
				return handleHttpMessageNotWritable(
						(HttpMessageNotWritableException) ex, request, response, handler);
			}
			else if (ex instanceof MethodArgumentNotValidException) {
				return handleMethodArgumentNotValidException(
						(MethodArgumentNotValidException) ex, request, response, handler);
			}
			else if (ex instanceof MissingServletRequestPartException) {
				return handleMissingServletRequestPartException(
						(MissingServletRequestPartException) ex, request, response, handler);
			}
			else if (ex instanceof BindException) {
				return handleBindException((BindException) ex, request, response, handler);
			}
			else if (ex instanceof NoHandlerFoundException) {
				return handleNoHandlerFoundException(
						(NoHandlerFoundException) ex, request, response, handler);
			}
			else if (ex instanceof AsyncRequestTimeoutException) {
				return handleAsyncRequestTimeoutException(
						(AsyncRequestTimeoutException) ex, request, response, handler);
			}
		}
		catch (Exception handlerEx) {
			if (logger.isWarnEnabled()) {
				logger.warn("Failure while trying to resolve exception [" + ex.getClass().getName() + "]", handlerEx);
			}
		}
		return null;
	}

5.自定义实现 HandlerExceptionResolver 处理异常,可以作为默认的全局异常处理规则

@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(521,"I love you !");
        } catch (IOException e) {
            e.printStackTrace();
        }
        return new ModelAndView();
    }
}

详解SpringBoot异常处理流程及原理

ErrorViewResolver 实现自定义处理异常。

(1)底层调用response.sendError时 ,error请求就会默认转给basicErrorControllerBasicErrorController专门来处理/error请求,适配4xx.html或者5xx.html页面

(2)如果异常没有任何解析器能处理,tomcat底层 也会调用response.sendErrorerror请求就会默认转给basicErrorControllerBasicErrorController专门来处理/error请求,适配4xx.html或者5xx.html页面。

(3)basicErrorController 要去的页面地址是由 ErrorViewResolver这个错误视图解析器决定的,即适配4xx.html或者5xx.html页面。

到此这篇关于详解SpringBoot异常处理流程及原理的文章就介绍到这了,更多相关SpringBoot异常处理流程及原理内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Java/Android 相关文章推荐
springboot中一些比较常用的注解总结
Jun 11 Java/Android
Java中使用Filter过滤器的方法
Jun 28 Java/Android
java设计模式--三种工厂模式详解
Jul 21 Java/Android
Log4j.properties配置及其使用
Aug 02 Java/Android
Java移除无效括号的方法实现
Aug 07 Java/Android
深入解读Java三大集合之map list set的用法
Nov 11 Java/Android
使用HttpSessionListener监听器实战
Mar 17 Java/Android
Mybatis-Plus进阶分页与乐观锁插件及通用枚举和多数据源详解
Mar 21 Java/Android
Android在Sqlite3中的应用及多线程使用数据库的建议
Apr 24 Java/Android
Java死锁的排查
May 11 Java/Android
多线程Spring通过@Scheduled实现定时任务
May 25 Java/Android
阿里面试Nacos配置中心交互模型是push还是pull原理解析
Jul 23 Java/Android
Java新手教程之ArrayList的基本使用
Java各种比较对象的方式的对比总结
Jun 20 #Java/Android
Java Optional<Foo>转换成List<Bar>的实例方法
Jun 20 #Java/Android
详解Java实践之适配器模式
Java数据结构之链表相关知识总结
详解Java实践之抽象工厂模式
Java框架入门之简单介绍SpringBoot框架
You might like
php多重接口的实现方法
2015/06/20 PHP
超强多功能php绿色集成环境详解
2017/01/25 PHP
js通过元素class名字获取元素集合的具体实现
2014/01/06 Javascript
构造函数+原型模式构造js自定义对象(最通用)
2014/05/12 Javascript
jquery中show()、hide()和toggle()用法实例
2015/01/15 Javascript
jQuery实现dialog设置focus焦点的方法
2015/06/10 Javascript
js基础知识(公有方法、私有方法、特权方法)
2015/11/06 Javascript
js实现微信分享代码
2020/10/11 Javascript
jQuery+canvas实现简单的球体斜抛及颜色动态变换效果
2016/01/28 Javascript
js实现HashTable(哈希表)的实例分析
2016/11/21 Javascript
教你快速搭建Node.Js服务器的方法教程
2017/03/30 Javascript
vue-axios使用详解
2017/05/10 Javascript
javascript基于牛顿迭代法实现求浮点数的平方根【递归原理】
2017/09/28 Javascript
JS加密插件CryptoJS实现的DES加密示例
2018/08/16 Javascript
JS使用Date对象实时显示当前系统时间简单示例
2018/08/23 Javascript
vue实现导航标题栏随页面滚动渐隐渐显效果
2020/03/12 Javascript
在MAC上搭建python数据分析开发环境
2016/01/26 Python
Python实现购物车功能的方法分析
2017/11/10 Python
Python函数any()和all()的用法及区别介绍
2018/09/14 Python
详解解决Python memory error的问题(四种解决方案)
2019/08/08 Python
python怎么删除缓存文件
2020/07/19 Python
Python3+PyCharm+Django+Django REST framework配置与简单开发教程
2021/02/16 Python
CSS3 animation实现逐帧动画效果
2016/06/02 HTML / CSS
Doyoueven官网:澳大利亚健身服饰和配饰品牌
2019/03/24 全球购物
某公司C#程序员面试题笔试题
2014/05/26 面试题
初始化了一个没有run()方法的线程类,是否会出错?
2014/03/27 面试题
如何防止同一个帐户被多人同时登录
2013/08/01 面试题
公司道歉信范文
2014/01/09 职场文书
农村文化活动总结
2014/08/28 职场文书
2014年全国法制宣传日宣传活动方案
2014/11/02 职场文书
给老婆的道歉信
2015/01/20 职场文书
明星邀请函
2015/02/02 职场文书
2015年银行柜员工作总结报告
2015/04/01 职场文书
2016国庆节67周年寄语
2015/12/07 职场文书
读《皮囊》有感:理解是对他人的最大的善举
2019/11/14 职场文书
详解如何在Canvas中添加事件的方法
2021/04/17 Javascript