6.1 默认规则
1.默认情况下,Spring Boot提供
/error
处理所有错误的映射。
2.对于机器客户端,它将生成JSON响应,其中包含错误,HTTP状态和异常消息的详细信息;
对于浏览器客户端,响应一个“ whitelabel”错误视图,以HTML格式呈现相同的数据。
3.要对其进行自定义,添加
View
视图解析为error
。
4.要完全替换默认行为,可以实现
ErrorController
并注册该类型的Bean定义,或添加ErrorAttributes类型的组件
以使用现有机制替换其内容。
5.error/下的4xx,5xx页面会被自动解析。
6.2 定制错误处理逻辑
1)、自定义错误页
error/404.html error/5xx.html
有精确的错误状态码页面就匹配精确,没有就找 4xx.html;如果都没有就触发白页;
触发点:
@Controller
public class TableController {
//触发400 Bad Request
@GetMapping("/basic_table")
public String basic_table(Model model, @RequestParam("a")int a){
//模拟数据
List<User> users
= Arrays.asList(new User("zhangsan", "123456"),
new User("lisi", "123444"),
new User("haha", "aaaaa"),
new User("hehe ", "aaddd"));
model.addAttribute("users",users);
return "table/basic_table";
}
}
2)、@ControllerAdvice+@ExceptionHandler全局异常处理器
底层是 ExceptionHandlerExceptionResolver 支持的
@Slf4j
@ControllerAdvice //增强控制器
public class GlobalExceptionhandler {
//定义一个专门处理"数学异常"的处理器
@ExceptionHandler({ArithmeticException.class})
public String handlerArithException(Exception e){
log.error("异常:{}",e.getMessage());
return "login";
}
}
触发点:
@Controller
public class TableController {
@GetMapping("/basic_table")
public String basic_table(Model model){
//模拟数据
List<User> users
= Arrays.asList(new User("zhangsan", "123456"),
new User("lisi", "123444"),
new User("haha", "aaaaa"),
new User("hehe ", "aaddd"));
//触发数学异常
int i = 10/0;
model.addAttribute("users",users);
return "table/basic_table";
}
}
3)、@ResponseStatus+自定义异常
底层是 **ResponseStatusExceptionResolver支持的 **
把responsestatus注解的信息底层调用 response.sendError(statusCode, resolvedReason);
tomcat发送的/error
//自定义异常 绑定响应错误状态码
@ResponseStatus(value = HttpStatus.BAD_REQUEST)
public class CustomException extends RuntimeException {
public CustomException() {
super();
}
public CustomException(String message) {
super(message);
}
}
触发点:
@Controller
public class TableController {
@GetMapping("/basic_table")
public String basic_table(Model model){
//模拟数据
List<User> users
= Arrays.asList(new User("zhangsan", "123456"),
new User("lisi", "123444"),
new User("haha", "aaaaa"),
new User("hehe ", "aaddd"));
//触发自定义异常
if(users.size()>3){
throw new CustomException("太多值");
}
model.addAttribute("users",users);
return "table/basic_table";
}
}
4)、Spring底层的异常,如参数类型转换异常
底层是DefaultHandlerExceptionResolver支持的,处理框架底层的异常(例如:400 缺失请求参数)
response.sendError(HttpServletResponse.SC_BAD_REQUEST, ex.getMessage());
此次请求立即结束,tomcat发送的/error,被BasicErrorController控制器处理。
tomcat原生的错误页:
5)、自定义实现 HandlerExceptionResolver 处理异常
可以作为默认的全局异常处理规则
//设置优先级,数字越小优先级越高
@Order(value = Ordered.HIGHEST_PRECEDENCE)
@Component
public class CustomHandlerExceptionResolver implements HandlerExceptionResolver {
@Override
public ModelAndView resolveException(HttpServletRequest request,
HttpServletResponse response,
Object handler,
Exception ex) {
try {
response.sendError(422,"自定义的错误");
} catch (IOException e) {
e.printStackTrace();
}
return new ModelAndView();
}
}
6)、ErrorViewResolver 实现自定义处理异常
1.只要执行response.sendError() ,error请求就会转给controller;
2.出现的异常没有任何人能处理,tomcat底层也是 response.sendError(),error请求就会转给controller;
3.BasicErrorController 要跳转的页面地址是由 ErrorViewResolver视图解析器解析的 ;
6.3 【源码分析】异常处理自动配置原理
-
ErrorMvcAutoConfiguration 自动配置异常处理规则
-
- 容器中的组件:类型:DefaultErrorAttributes -> id:errorAttributes
-
-
- public class DefaultErrorAttributes implements ErrorAttributes, HandlerExceptionResolver
- DefaultErrorAttributes:定义错误页面中可以包含哪些数据。
-
-
- **容器中的组件:类型:**BasicErrorController --> id:basicErrorController(json+白页 适配响应)
-
-
- 处理默认 /error 路径的请求;页面响应 new ModelAndView(“error”, model);
- 容器中有组件 View->id是error;(响应默认错误页)
- 容器中放组件 BeanNameViewResolver(视图解析器);按照返回的视图名作为组件的id去容器中找View对象。
-
-
- **容器中的组件:**类型:**DefaultErrorViewResolver -> id:**conventionErrorViewResolver
-
-
- 如果发生错误,会以HTTP的状态码 作为视图页地址(viewName),找到真正的页面
- error/404、5xx.html
-
如果想要返回页面;就会找error视图【StaticView】。(默认是一个白页)
6.4 【源码分析】异常处理步骤流程
1、执行目标方法,目标方法运行期间有任何异常都会被catch、而且标志当前请求结束;并且用 dispatchException
2、进入视图解析流程(页面渲染)
processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException);
3、mv = processHandlerException;处理handler发生的异常,处理完成返回ModelAndView;
- 1、遍历所有的 **handlerExceptionResolvers,看谁能处理当前异常【**HandlerExceptionResolver处理器异常解析器】
- -2、系统默认的 异常解析器;
-
- 1、DefaultErrorAttributes先来处理异常。把异常信息保存到rrequest域,并且返回null;
- 2、默认没有任何人能处理异常,所以异常会被抛出
-
-
- 1、如果没有任何人能处理最终底层就会发送 /error 请求。会被底层的BasicErrorController处理
- 2、解析错误视图;遍历所有的 ErrorViewResolver 看谁能解析。
-
-
- 3、默认的 DefaultErrorViewResolver ,作用是把响应状态码作为错误页的地址,error/500.html
- 4、模板引擎最终响应这个页面 error/500.html
-