1. 前言
统一的异常处理对于应用的重要性不言而喻。今天我们来介绍一下 Spring 如何来进行统一的 Rest 异常处理。同时我们也会简单比较一下它们之间的优劣。
2. @Controller 结合 @ExceptionHandler
在控制器中声明一个方法然后用
@ExceptionHandler
注解标记即可:
@Controller
@RequestMapping("/test")
public class TestController {
@RequestMapping("/err")
@ResponseBody
public Object demo1(){
int i = 1 / 0;
return new Date();
}
@ExceptionHandler({RuntimeException.class})
public ModelAndView fix(Exception ex){
System.out.println(ex.getMessage());
return new ModelAndView("error",new ModelMap("ex",ex.getMessage()));
}
}复制代码
优点:
- 优先级最高。
-
@ExceptionHandler
标记的方法返回值类型支持多种。可以是视图,也可以是json
等。
缺点:
-
一个
Controller
中的@ExceptionHandler
注解上的异常类型不能出现相同的,否则运行时抛异常。 - 需要显式的声明处理的异常类型。
-
作用域仅仅是该
Controller
并不是真正意义上的全局异常。如果要想作用于全局需要将其放入所有控制器的父类中。
3. @ControllerAdvice 结合 @ExceptionHandler
这是
2.
的改进型,通过定义
@ControllerAdvice
类并在方法上标记
@ExceptionHandler
,达到了全局异常处理的目的:
@ControllerAdvice
public class TestController {
@ExceptionHandler({RuntimeException.class})
public ModelAndView fix(Exception ex){
System.out.println(ex.getMessage());
return new ModelAndView("error",new ModelMap("ex",ex.getMessage()));
}
}复制代码
优点:
- 全局的异常处理。
- 完全控制响应的主体以及状态码
- 将多个异常映射到同一方法,以一起处理,并且它充分利用了更新的 Restful ResponseEntity 响应
缺点:
-
一个
Controller
中的@ExceptionHandler
注解上的异常类型不能出现相同的,否则运行时抛异常。 - 需要显式的声明处理的异常类型。
一般情况下也建议使用该方式进行异常处理。大多数情况下都是兼容的。
4. HandlerExceptionResolver 接口
实现
HandlerExceptionResolver
接口,这里我们继承其抽象实现
AbstractHandlerExceptionResolver
:
@Component
public class RestResponseStatusExceptionResolver extends AbstractHandlerExceptionResolver {
@Override
protected ModelAndView doResolveException(
HttpServletRequest request,
HttpServletResponse response,
Object handler,
Exception ex) {
try {
if (ex instanceof IllegalArgumentException) {
return handleIllegalArgument((IllegalArgumentException) ex, response, handler);
}
//todo more exception
} catch (Exception handlerException) {
//todo
}
return null;
}
private ModelAndView
handleIllegalArgument(IllegalArgumentException ex, HttpServletResponse response)
throws IOException {
response.sendError(HttpServletResponse.SC_CONFLICT);
String accept = request.getHeader(HttpHeaders.ACCEPT);
//todo more response
return new ModelAndView();
}
}复制代码