这里写目录标题
1 引言
ControllerAdvice(RestControllerAdvice ) ,ControllerAdvice 是无法处理过滤器中的异常的。
下面介绍的方式,其思想早在 Struts 拦截器时代就存在了。有兴趣的可以去看看 Sturts 拦截器组,第一个就是 exception 拦截器。本质上是借助过滤器栈,将异常处理的过滤器放在第一个位置。
引用一张图
SpringBoot中,通过@ControllerAdvice+@ExceptionHandler实现的全局异常捕获类,因为Filter是在Controller层之前的,所以只能捕获Controller层的异常,无法捕获filter抛出的异常。
2 springboot的全局异常捕获(controller层)
全局异常捕获代码
@RestControllerAdvice
public class ExceptionHandle {
private final Logger logger = LoggerFactory.getLogger(getClass());
@ExceptionHandler(value = Exception.class)
public ResultVo<Object> handle(Exception e) {
logger.error(e.getMessage(), e);
if (e instanceof MyException) {
MyException myException = (MyException) e;
return ResultUtil.error(myException);
} else {
return ResultUtil.error(ResultEnum.UNKNOWN_ERROR.getCode(), e.getMessage());
}
}
/**
* 方法参数校验
* 由于spring捕获异常的问题,导致此异常的编码格式无法修改为UTF-8,
* 故不使用RestControlle的ResponseBody,自己实现httpServletResponse的IO。
*/
@ExceptionHandler(MethodArgumentNotValidException.class)
public void handleMethodArgumentNotValidException(HttpServletResponse httpServletResponse, MethodArgumentNotValidException e) {
logger.error(e.getMessage(), e);
ResultVo<Object> resultVo = new ResultVo<>();
resultVo.setCode(ResultEnum.VALIDATION_ERROR.getCode());
resultVo.setMsg(Objects.requireNonNull(e.getBindingResult().getFieldError()).getDefaultMessage());
try {
ResultUtil.sendResponse(httpServletResponse ,resultVo);
} catch (IOException ioException) {
ioException.printStackTrace();
}
}
@ExceptionHandler(ValidationException.class)
public ResultVo<Object> handleValidationException(ValidationException e) {
logger.error(e.getMessage(), e);
return ResultUtil.error(ResultEnum.VALIDATION_ERROR.getCode(), e.getCause().getMessage());
}
@ExceptionHandler(ConstraintViolationException.class)
public ResultVo<Object> handConstraintViolationException(ConstraintViolationException e) {
logger.error(e.getMessage(), e);
return ResultUtil.error(ResultEnum.VALIDATION_ERROR.getCode(), e.getMessage());
}
}
3 解决方式1:对filter的异常进行catch转发到controller层处理
在filter中进行try catch
try {
...
chain.doFilter(request, response);
} catch (Exception e) {
request.setAttribute("exception", e);
request.getRequestDispatcher("/error").forward(request, response);
}
在controller中捕获
@PostMapping("/error")
public ResponseVO throwException(HttpServletRequest request) throws Exception {
throw (Exception) request.getAttribute("exception");
}
4 解决方式2:对filter的异常进行catch然后直接返回前端数据进行return
try {
...
chain.doFilter(request, response);
} catch (Exception e) {
sendResponse...
}