默认情况下的异常现象
创建一个接口 (接口需要传递参数key)
@RestController
@RequestMapping("/exception")
public class ExceptionController {
@GetMapping("/accept")
public String acceptKey(@RequestParam("key") String key) {
return key;
}
}
访问链接(不传递参数key,使得抛出异常) http://localhost:8080/exception/accept
在浏览器中的现象
data:image/s3,"s3://crabby-images/27603/27603d3011ac48706b4e2a8f445ac19a7c67ba16" alt=""
在 Postman 中的现象
data:image/s3,"s3://crabby-images/a0aec/a0aeccf95610d75a7c10da127946aad59ccd52b6" alt=""
小结
在浏览器中返回一个 html 页面,在 Postman 中返回一个 json 数据
解决方案
在默认静态资源路径下创建 error 子文件夹,并创建文件 400.html
默认静态资源路径如下:
- classpath:/META-INF/resources/
- classpath:/resources/
- classpath:/static/
- classpath:/public/
400.html 明细如下所示:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
<h1>404</h1>
</body>
</html>
data:image/s3,"s3://crabby-images/bb304/bb304598cce651bc4a1c5ae09697db7c36563ddf" alt=""
**再次在浏览器下访问链接 :**http://localhost:8080/exception/accept
data:image/s3,"s3://crabby-images/38614/38614c71ca6d3ed22d0f44c58dd7878cfcba3def" alt=""
PS:错误码需要和 html 名字一致,或者将 html 改成 4xx、5xx,这样以 4 开头的错误码就会跳转到 4xx.html,以 5 开头的错误码就会跳转到 5xx.html
再次在 Postman 中**访问链接 :**http://localhost:8080/exception/accept
data:image/s3,"s3://crabby-images/55593/55593b09055030631590f92f34b650a2cb1a45f6" alt=""
好像并没有起作用,我们再尝试其他方法
创建 GlobalExceptionHandler,处理全局异常
创建实体类 ExceptionInfo
@Data
@NoArgsConstructor
@AllArgsConstructor
public class ExceptionInfo {
private String msg;
}
创建全局异常处理配置类 GlobalExceptionHandler
@ControllerAdvice
public class GlobalExceptionHandler {
@ExceptionHandler(value = Exception.class)
@ResponseBody
public ExceptionInfo resolverException(Exception exception) {
ExceptionInfo exceptionInfo = new ExceptionInfo();
exceptionInfo.setMsg(exception.getMessage());
return exceptionInfo;
}
}
**访问链接 :**http://localhost:8080/exception/accept
data:image/s3,"s3://crabby-images/623de/623de71b0f056add29c87d2c14826488a696023f" alt=""
源码解析
ErrorMvcAutoConfiguration
data:image/s3,"s3://crabby-images/a227e/a227e8e9283748179d7537654859d3b9a36deb9e" alt=""
data:image/s3,"s3://crabby-images/0feda/0feda51a09fec740020a8f79482d2bbe1f377643" alt=""
ErrorMvcAutoConfiguration 会定义一个类型为 BasicErrorController 的 Bean。BasicErrorController 中定义了两个接口方法:errorHtml、error ,其中 errorHtml 方法只对 request 的 accept 能兼容 text/html 的请求有效,error 方法则可以认为是一个兜底方法,它对 request 的 accept 没有要求
当我们的请求抛出异常,会转发到这个接口(/error) ,如果这个默认的 URI(/error) 和我们的项目有冲突,我们可以在配置文件中定义 server.error.path | error.path 来修改默认值。
StandardHostValve#custom (请求转发)
data:image/s3,"s3://crabby-images/dd582/dd582229fa080ec7b6d23693107a5e15079f49be" alt=""
DispatcherServlet#processDispatchResult
data:image/s3,"s3://crabby-images/ea625/ea62578fb7b40d0ab06a734bacebd7808bb7e6b2" alt=""
我们需要关注两个方法 processHandlerException、render。
case1:访问一个不存在的接口或者不存在的文件
这种情况 exception 和 mv(ModelAndView)都为 null,所以既不会执行 processHandlerException 方法,也不会执行 render 方法,然后转发到 /error 接口。
在 Postman 中发请求,Accept 默认为 */*,在浏览器中发请求 Accept 如下所示:
data:image/s3,"s3://crabby-images/6e42b/6e42babd501b34369e93f9adb95f8a1298bd72a3" alt=""
根据一定的规则,在 Postman 中默认转发到 error 方法,在浏览器中默认转发到 errorHtml 方法
AbstractHandlerMethodMapping#lookupHandlerMethod (请求映射规则)
data:image/s3,"s3://crabby-images/5e3e4/5e3e4d28d11b19a461357af8c203b535590faa42" alt=""
case1.1 转发到 error 方法
data:image/s3,"s3://crabby-images/0a317/0a3174601c9545c1f936052f432c6e4a0c87fa93" alt=""
data:image/s3,"s3://crabby-images/42f90/42f9080208071d29bfe2bd712de510728c690260" alt=""
data:image/s3,"s3://crabby-images/dc15d/dc15d4c4d9033d65d7fb065e6c6e288a2c93ce1e" alt=""
data:image/s3,"s3://crabby-images/dc96f/dc96f6da785652ee49295697fc0c426488f79f02" alt=""
data:image/s3,"s3://crabby-images/58b27/58b27c1bf2f53aaf917e849bb80575d9a80314d1" alt=""
该方法会构建一个 map 对象,设置 timestamp、status、error、path 等信息并响应
case1.2 转发到 errorHtml 方法
data:image/s3,"s3://crabby-images/de2e8/de2e8a76de078073b64c255e631c90e1673e2a56" alt=""
data:image/s3,"s3://crabby-images/45f37/45f37972196396e0705b1efa1d2f7705f3618612" alt=""
默认情况下 errorViewResolvers 只有一个,类型为 DefaultErrorViewResolver,它是在 ErrorMvcAutoConfiguration 内部类 DefaultErrorViewResolverConfiguration 中定义的,相关源码如下:
data:image/s3,"s3://crabby-images/91c67/91c67e97920e1ac1fbf46595307acceedc42585d" alt=""
DefaultErrorViewResolver#resolveErrorView
data:image/s3,"s3://crabby-images/70711/70711898d77f2f37cdb2fb1b97ac58c6a6ec5c35" alt=""
当我们访问一个不存在的链接,viewName 为 404,errorViewName 为 error/404,如果 TemplateAvailabilityProviders 的 getProvider 方法返回一个非 null 对象,则返回一个 ModelAndView 对象
TemplateAvailabilityProviders#getProvider
data:image/s3,"s3://crabby-images/0b563/0b56329ddc4ff56229b22ba293238c01c993416b" alt=""
如果 TemplateAvailabilityProvider 的 isTemplateAvailable 方法返回 true,则返回当前 TemplateAvailabilityProvider 对象,即最终会返回一个 ModelAndView 对象
data:image/s3,"s3://crabby-images/13397/133979d2734eeb121a59f641548634f768a27b9f" alt=""
根据 SpringBoot 的自动配置,容器中存在五个 TemplateAvailabilityProvider ,我们来看一下 ThymeleafTemplateAvailabilityProvider 的 isTemplateAvailable 方法。
data:image/s3,"s3://crabby-images/888a3/888a3ec4564b6b8d044c01feebad527702686168" alt=""
data:image/s3,"s3://crabby-images/f418d/f418d5904fa4c8da09757bc94db2323925459f90" alt=""
即默认情况下,如果我们的环境中,存在指定的类(org.thymeleaf.spring5.SpringTemplateEngine),并且资源 classpath:/templates/error/404.html 存在,则返回一个 ModelAndView 对象
可以导入下方所示的依赖
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-thymeleaf</artifactId>
<version>2.6.13</version>
</dependency>
DefaultErrorViewResolver#resolveResource
如果 TemplateAvailabilityProviders 的 getProvider 方法返回 null,则继续调用 resolveResource 方法。该方法会遍历 staticLocations,判断这些默认静态文件路径下是否存在相关文件(是否存在error/404.html),如果存在则构建一个 HtmlResourceView 对象,staticLocations 明细列表如下:
- classpath:/META-INF/resources/
- classpath:/resources/
- classpath:/static/
- classpath:/public/
DefaultErrorViewResolver#resolveErrorView (2)
data:image/s3,"s3://crabby-images/0179e/0179e6f34ba1fca8373cdd92ef3e0e67349551ad" alt=""
data:image/s3,"s3://crabby-images/af248/af24885166a9f9e52b1d91b9398a49f145036d2e" alt=""
如果错误状态码是以 4 或 5 开头,则以 viewName 为 4xx,errorViewName 为 error/4xx,再执行一遍上述的流程
BasicErrorController#errorHtml(2)
即不管我们有没有在当前环境中找到指定文件,都会返回一个 ModelAndView 对象,如果存在以下资源,viewName 则不为 error:
- classpath:/templates/error/404.html => view:error/404
- classpath:/META-INF/resources/404.html => view:HtmlResourceView
- classpath:/resources/404.html => view:HtmlResourceView
- classpath:/static/404.html => view:HtmlResourceView
- classpath:/public/404.html => view:HtmlResourceView
- classpath:/templates/error/4xx.html => view:error/4xx
- classpath:/META-INF/resources/4xx.html => view:HtmlResourceView
- classpath:/resources/4xx.html => view:HtmlResourceView
- classpath:/static/4xx.html => view:HtmlResourceView
- classpath:/public/4xx.html => view:HtmlResourceView
- 其他 => view:error
DispatcherServlet#processDispatchResult(2)
data:image/s3,"s3://crabby-images/cce6d/cce6d1c8071fb11ba98bf481024d1c5a8aa2ff36" alt=""
第二次进入 DispatcherServlet 的 processDispatchResult 方法,此时 mv 不等于null,则进入render 方法
data:image/s3,"s3://crabby-images/d7b2b/d7b2b2ffb4a9869899087d107f442e50d61f5985" alt=""
如果 viewName 为 error/404、error/4xx、error 则执行 resolveViewName 方法
data:image/s3,"s3://crabby-images/372d3/372d30c2f298662ff430ba0c68ad937d8a4f62e9" alt=""
data:image/s3,"s3://crabby-images/71dd9/71dd90804a3db3bd45acaef9819ff41e79f99c6a" alt=""
一共有5个resolver,我们只需要关注 ContentNegotiatingViewResolver 的 resolveViewName 方法
data:image/s3,"s3://crabby-images/4b8c8/4b8c88e122511f81ddc4b57a716938ed5637bd0f" alt=""
data:image/s3,"s3://crabby-images/e9254/e9254c3d701bd45c2fc11a471d3d9d4f64b656e9" alt=""
data:image/s3,"s3://crabby-images/7132d/7132dfe0fe10c64094a4dc5988c4f651b31e8317" alt=""
ContentNegotiatingViewResolver 的内部属性 viewResolvers 有其他四个 resolvers,即 ContentNegotiatingViewResolver 相当于一个大管家,具体还是由其他四个 resolvers 处理,我们简要分析一下 beanNameViewResolver
data:image/s3,"s3://crabby-images/704a6/704a6b1a50a8fcfe3d2fce83f724b13345c88493" alt=""
ErrorMvcAutoConfiguration 的内部类 WhitelabelErrorViewConfiguration 会定义两个bean,其中一个 beanName 为 error,类型为 View,另一个 beanName 为 beanNameViewResolver,类型为 BeanNameViewResolver
data:image/s3,"s3://crabby-images/47aff/47affe67a17501f80d983b61e994a52132fd742f" alt=""
即 BeanNameViewResolver 的 resolveViewName 方法会返回一个类型为 StaticView 的 View
View#render
StaticView#render
data:image/s3,"s3://crabby-images/aad9b/aad9b4893da4d43acca3aedb6e6e37252ff87c42" alt=""
我们可以看到 StaticView 的 render 方法就是我们经常看到的页面
StaticView#render
HtmlResourceView#render
HtmlResourceView 的 render 方法就是将指定资源用流写出去
case2:访问一个存在的接口且抛出异常
DispatcherServlet#processHandlerException
data:image/s3,"s3://crabby-images/0dfed/0dfed23847307e8914cdd717a35181731dfb653b" alt=""
data:image/s3,"s3://crabby-images/b423e/b423e7a86e09afcde23b98f44d6470a91c45051e" alt=""
一共有两个 HandlerExceptionResolver,其中一个类型为 DefaultErrorAttributes,DefaultErrorAttributes 的 resolveException 方法比较简单,主要是给 request 赋值,我们主要关注 HandlerExceptionResolverComposite 的 resolveException 方法
data:image/s3,"s3://crabby-images/af544/af544f7f1a909dd8e456ad0e34913739cbd4d2ff" alt=""
HandlerExceptionResolverComposite#resolveException
data:image/s3,"s3://crabby-images/ab63d/ab63d31a8bdc656173d76bf11e1f731b14374ee9" alt=""
data:image/s3,"s3://crabby-images/71626/71626ad7f5350a3f7c7641b4db825f27c9a346af" alt=""
一共有三个 HandlerExceptionResolver,我们主要分析一下 ExceptionHandlerExceptionResolver的 resolveException 方法
ExceptionHandlerExceptionResolver#resolveException
data:image/s3,"s3://crabby-images/fc738/fc738b20bf9b609b5d995f50348b4d6aea04e3be" alt=""
data:image/s3,"s3://crabby-images/3322a/3322a07b1e019e170a52e4ab4d9c760ef1bc4016" alt=""
data:image/s3,"s3://crabby-images/a8ce9/a8ce951468040cbbe10bd10f4b30ad5d0757a85c" alt=""
最终会执行类上标记 @ControllerAdvice 注解,方法上标记 @ExceptionHandler 注解的符合条件的方法,就是我们在【解决方案】的 GlobalExceptionHandler#resolverException 方法
如何选择 ServletInvocableHandlerMethod
data:image/s3,"s3://crabby-images/3a049/3a04952f4947204ca9a52fde32a5f96c79d18ca1" alt=""
遍历 exceptionHandlerAdviceCache 对象,通过 ExceptionHandlerMethodResolver 的resolveMethod 方法,获取一个 method 对象,并将其封装成 ServletInvocableHandlerMethod 对象
isApplicableToBeanType
data:image/s3,"s3://crabby-images/339a1/339a194608e8be3cd8481aeba1c072ea52757268" alt=""
以下四种情况,@ControllerAdvice注解生效:
- 什么都没有配置
- Controller所在的类路径以配置的 basePackages 开头
- Controller类型是指定的类或是其子类
- Controller上含有指定注解
resolveMethod
data:image/s3,"s3://crabby-images/200f1/200f1c2a28b0e895e4239e5a8b8c142d7e668cf9" alt=""
遍历 mappedMethods 对象,如果存在多个 @ExceptionHandler 标注的方法,则选择一个优先级最高的
ExceptionHandlerExceptionResolver 的 exceptionHandlerAdviceCache 是如何赋值的
data:image/s3,"s3://crabby-images/a38d1/a38d17f8434ae8401abac8b84e32567959b8ace6" alt=""
ExceptionHandlerExceptionResolver 继承 InitializingBean 接口,所以 bean 在实例化的过程中会执行其 afterPropertiesSet 方法
ExceptionHandlerExceptionResolver#afterPropertiesSet
data:image/s3,"s3://crabby-images/92c88/92c8817dadab9aae3d1e5ddc21a8dc83c0727b7b" alt=""
data:image/s3,"s3://crabby-images/69090/6909089fe13734678abbd8f55764e9629bb7359f" alt=""
如果 bean 上存在 @ControllerAdvice 注解,则构建一个 ControllerAdviceBean 对象
data:image/s3,"s3://crabby-images/c8d34/c8d34971917e6c30292b4ef2ef63079716aad312" alt=""
循环遍历查找出来的 adviceBeans,每存在一个 ControllerAdviceBean,则构建一个ExceptionHandlerMethodResolver 并将其 put 到 exceptionHandlerAdviceCache 中
ExceptionHandlerMethodResolver的实例化(给 mappedMethods 赋值)
data:image/s3,"s3://crabby-images/a4577/a45775da321dfe1cefb8fc2052a316b0da4a2ff7" alt=""
data:image/s3,"s3://crabby-images/9b511/9b511eedfce98f1bc237f52098f63c02e5438bb2" alt=""
如果方法存在 @ExceptionHandler 注解,则给 mappedMethods 赋值
PS : 如果 @ExceptionHandler 注解标注的方法也抛出异常,则使用 case1 做兜底