本文介绍了SpringBoot中全局异常处理的局限,即无法捕获Filter层的异常。提供了两种解决方案:一是通过Filter内部捕获异常并转发到Controller处理,二是直接在Filter内处理异常并返回前端数据。详细展示了相关代码实现。 摘要由CSDN通过智能技术生成

这里写目录标题

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...
				
由于项目是前后端分离,且使用的shiro+jwt的框架,需要自定义一个jwtFilter来拦截请求并进行token的验证,这里会出现各种token的异常,所以需要捕获一下,但是全局异常处理@ControllerAdvice不能捕捉自定义拦截器的异常,所以这里使用继承BasicErrorController类来处理异常 要写一个构造器 public ErrorController() { super(new DefaultErrorAttributes(), new ErrorProperties()
此处我采用了JwtFilter 在每个请求之前 拦截器都会对请求进行拦截 导致本来很好用的@ControllerAdvice不管用了 那么怎么办呢 可以使用response即可 方法参数没有 也可以直接注入 代码示例如下 @SneakyThrows @Override protected boolean isAccessAllowed(ServletRequest request, ServletResponse response, Object mappedValue) { import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation...
  紧接上一篇文章展开主题,上篇链接:https://blog.csdn.net/qq_42227281/article/details/106869400   本篇:Request 获取Post请求 body的参数   1、Springboot配置过滤器   需要注意的是,我用的是springboot,...
xxxxingy: kubectl delete -f https://raw.githubusercontent.com/projectcalico/calico/v3.29.0/manifests/tigera-operator.yaml 其他版本类似 java高级:利用Lambda表达式解决代码中硬编码的问题 废话真多,能不能打印方法名? 理解Java泛型的复杂写法<? super T>,<? extend T> Yigezxy: ? super T声明泛型可以更大,往里放数只能更小 ? extends T声明泛型可以更小,不允许放置 nginx日志中文显示为16进制问题解决 房子的屋顶: 建议这个 感觉看起来更明显 log_format main escape=json '{"realip":"$remote_addr","@timestamp":"$time_iso8601","host":"$http_host","request":"$request","req_body":"$request_body","status":"$status","size":$body_bytes_sent,"ua":"$http_user_agent","cookie":"$http_cookie","req_time":"$request_time","uri":"$uri","referer":"$http_referer","xff":"$http_x_forwarded_for","ups_status":"$upstream_status","ups_addr":"$upstream_addr","ups_time":"$upstream_response_time"}'; Linux 实例常用内核参数介绍—容器访问外部网络之ip_forward数据包转发 m0_74268689: 真的真好,帮我答疑解惑了谢谢^ω^