springboot中过滤器异常无法被全局异常捕获

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…
}

  • 0
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在 Spring Boot ,可以通过自定义全局异常处理器来捕获和处理应用程序异常。以下是一种常见的实现方式: 首先,创建一个全局异常处理器类,例如 `GlobalExceptionHandler`,并使用 `@ControllerAdvice` 注解将其标记为全局异常处理器。在类可以定义多个方法来处理不同类型的异常。 ```java @ControllerAdvice public class GlobalExceptionHandler { @ExceptionHandler(Exception.class) public ResponseEntity<String> handleException(Exception ex) { // 处理异常逻辑 return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body("Internal Server Error"); } @ExceptionHandler(MyCustomException.class) public ResponseEntity<String> handleMyCustomException(MyCustomException ex) { // 处理自定义异常逻辑 return ResponseEntity.status(HttpStatus.BAD_REQUEST).body(ex.getMessage()); } // 其他异常处理方法... } ``` 在以上代码,`@ExceptionHandler` 注解用于指定每个方法要处理的异常类型。在方法,你可以根据具体需求编写异常处理逻辑,并返回合适的响应。 需要注意的是,全局异常处理器类需要被扫描到,可以通过将其所在的包作为扫描路径配置到 Spring Boot 的配置文件,或者使用 `@ComponentScan` 注解进行手动配置。 另外,还可以通过实现 `ErrorController` 接口来处理 Spring Boot 应用程序无法被 `GlobalExceptionHandler` 捕获的错误。 这样,当应用程序发生异常时,全局异常处理器会捕获并处理它们,返回相应的错误信息给客户端。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值