这里写自定义目录标题
- 1,Java Servlet规范
- 异常处理
- 触发异常
- 2,tomcat 异常处理实现
- tomcat请求处理流程
- 异常发生时核心处理类
- 3,springmvc 异常定制以及扩展
- 请求处理流程
- 使用@ControllerAdvice & @ExceptionHandler 配置全局异常处理器
- 未配置全局异常处理或者没有匹配到异常时的处理流程
- 以上处理过程中的未捕获异常,交给tomcat去处理异常
1,Java Servlet规范
异常处理
在web.xml中定义Exception和error-code的映射。
<!-- 根据状态码配置 -->
<error-page>
<error-code>404</error-code>
<location>/404.html</location>
</error-page>
<!-- 状态码为0时可以匹配所有的状态码 -->
<error-page>
<error-code>0</error-code>
<location>/error.html</location>
</error-page>
<!-- 根据异常类型配置 -->
<error-page>
<exception-type>java.lang.NullPointerException</exception-type>
<location>/error.html</location>
</error-page>
触发异常
HttpServletResponse.setError();
HttpServletResponse.sendError(status,msg);
2,tomcat 异常处理实现
tomcat请求处理流程
-
线程模型
-
请求处理流程
异常发生时核心处理类
当异常发生时,首先在StandardHostValve 中匹配,匹配的规则是
- 先根据Exception去找错误页面
- 再根据code去找错误页面
找到对应的页面后,dispatch到相应的页面,返回给前端进行展示。
如果没有匹配到相应的页面,tomcat使用以下两种Value进行异常处理。
ErrorReportValve
用于在界面上展示错误信息,也就是常见的错误提示:
StandardWrapperValve
收集执行Servlet过程中的错误信息,给ErrorReportValve进行展示。
3,springmvc 异常定制以及扩展
请求处理流程
使用@ControllerAdvice & @ExceptionHandler 配置全局异常处理器
用于处理 HandlerMethod(上图中 Handler) 执行过程中抛出的异常
未配置全局异常处理或者没有匹配到异常时的处理流程
- springboot在启动时,自动配置机制,使用 ErrorMvcAutoConfiguration 中注册:BasicErrorController
- 同时,注册code为0的ErrorPage
org.springframework.boot.autoconfigure.web.servlet.error.ErrorMvcAutoConfiguration.ErrorPageCustomizer#registerErrorPages
@Override
public void registerErrorPages(ErrorPageRegistry errorPageRegistry) {
ErrorPage errorPage = new ErrorPage(
this.dispatcherServletPath.getRelativePath(this.properties.getError().getPath()));
errorPageRegistry.addErrorPages(errorPage);
}
by the way:code为0的错误页面映射,tomcat中的处理逻辑如下:
//org.apache.catalina.valves.ErrorReportValve
protected void report(Request request, Response response, Throwable throwable) {
int statusCode = response.getStatus();
// Do nothing on a 1xx, 2xx and 3xx status
// Do nothing if anything has been written already
// Do nothing if the response hasn't been explicitly marked as in error
// and that error has not been reported.
if (statusCode < 400 || response.getContentWritten() > 0 || !response.setErrorReported()) {
return;
}
// If an error has occurred that prevents further I/O, don't waste time
// producing an error report that will never be read
AtomicBoolean result = new AtomicBoolean(false);
response.getCoyoteResponse().action(ActionCode.IS_IO_ALLOWED, result);
if (!result.get()) {
return;
}
ErrorPage errorPage = null;
if (throwable != null) {
errorPage = errorPageSupport.find(throwable);
}
if (errorPage == null) {
errorPage = errorPageSupport.find(statusCode);
}
if (errorPage == null) {
// Default error page
errorPage = errorPageSupport.find(0);
}
... //忽略部分代码
}
- 经过code为0的映射处理后,tomcat container组件会把请求dispatch到“/error”页面,最终找到BasicErrorController 处理器:
BasicErrorController 处理器的处理逻辑如下:
BasicErrorController 通过 Accept 头判断需要生成哪种 MediaType 的响应
3.1 如果要的不是 text/html,走 MessageConverter 流程
3.2 如果需要 text/html,走 mvc 流程,此时又分两种情况
a. 配置了 ErrorViewResolver,根据状态码去找 View
b. 没配置或没找到,用 BeanNameViewResolver 根据一个固定为 error 的名字找到 View,即所谓的 WhitelabelErrorView
以上处理过程中的未捕获异常,交给tomcat去处理异常
返回给tomcat的ErrorReportValve去处理,展示异常如前图所示。