SpringBoot对Filter过滤器中的异常进行全局处理

前言

今天处理拦截器中的异常时,遇到这样一个问题,我们希望在过滤器中对用户的请求进行判断,如果不符合要求直接抛出异常并在前端展示。但是如果我们直接在过滤器中throw一个异常时,尽管我们使用@ControllerAdvice和 @ExceptionHandler注解注册了全局异常处理器,但是前端是无法接收到的。

这是因为过滤器是在进入Servlet之前处理请求的,从注解名称也能看出来@ControllerAdvice是处理Controller层异常的,请求还没到Controller层,当然无法处理了。

那么如何将过滤器中的异常像Controller层一样进行全局处理呢?

一、全局异常处理

首先全局异常处理器还是不可少的,大概像下面这个样子

@ControllerAdvice
public class GlobalExceptionHandler {
    @ExceptionHandler(value = ServiceException.class)
    @ResponseBody
    public ResponseEntity<AjaxResponse> serviceExceptionHandler(ServiceException e, HttpServletRequest request){
        //...处理一些逻辑
        return new ResponseEntity<>(new AjaxResponse().failure(Constants.ErrorCode.SERVICE_ERROR,e.getMessage()), HttpStatus.BAD_REQUEST);
    }
}

二、处理过滤器中的异常

在我们自定义的Filter类中,使用HandlerExceptionResolver的resolveException方法来进行处理,如下面代码所示

@Component
public class MyFilter implements Filter {

    @Override
    public void init(javax.servlet.FilterConfig filterConfig) throws ServletException {
    }

    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
            //...进行一些逻辑处理
        	if (...) {
                HandlerExceptionResolver handlerExceptionResolver = SpringContextUtil.getBean("handlerExceptionResolver");
                handlerExceptionResolver.resolveException((HttpServletRequest) servletRequest, (HttpServletResponse) servletResponse,
                        null,new ServiceException("抛出过滤器全局异常!"));
                return;
            }
        filterChain.doFilter(servletRequest, servletResponse);
    }

    @Override
    public void destroy() {

    }
}

这样就能实现效果了:

在这里插入图片描述

看一下后端返回的响应数据如下:

在这里插入图片描述

接着我们说说HandlerExceptionResolver和它的resolveException方法的原理实现。

HandlerExceptionResolver 接口是 Spring MVC 提供的全局异常处理机制的一部分,它的原理基于 Spring MVC 框架的请求处理流程和异常处理机制。

当请求在 Spring MVC 中的某一组件中抛出异常时,异常会沿着请求处理的链路向上传播,直到找到能够处理异常的地方。通常,异常首先会被容器(例如,Servlet 容器)捕获,然后传递给 Spring MVC 框架。Spring MVC 框架会遍历已注册的异常处理器,调用它们的 resolveException 方法。每个异常处理器有机会检查异常类型,如果匹配,就执行自定义的异常处理逻辑。

在SpringBoot中我们通过 @ControllerAdvice 注解和 @ExceptionHandler注解注册了全局异常处理器,因此resolveException方法会将异常进行传播给我们自定义的异常处理器,最后就能进行全局异常处理了。

  • 4
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在Spring Boot项目,要将Filter过滤器声明为启动,有以下两种方式: 1. 使用@WebFilter注解 通过在Filter类上标记`@WebFilter`注解来声明Filter过滤器。这个注解是Servlet 3.0规范的一部分,并且Spring Boot支持它。在注解可指定该过滤器的URL模式以及过滤器在容器的执行顺序。 示例代码如下: ```java import javax.servlet.*; import javax.servlet.annotation.WebFilter; import java.io.IOException; @WebFilter(filterName = "myFilter", urlPatterns = "/*") public class MyFilter implements Filter { @Override public void init(FilterConfig filterConfig) throws ServletException { System.out.println("Filter init"); } @Override public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException { System.out.println("Filter doFilter"); filterChain.doFilter(servletRequest, servletResponse); } @Override public void destroy() { System.out.println("Filter destroy"); } } ``` 2. 使用FilterRegistrationBean 另一种方式是使用Spring Boot提供的FilterRegistrationBean类将Filter过滤器进行注册。 示例代码如下: ```java import org.springframework.boot.web.servlet.FilterRegistrationBean; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import javax.servlet.Filter; @Configuration public class MyFilterConfig { @Bean public FilterRegistrationBean<Filter> myFilter() { FilterRegistrationBean<Filter> registrationBean = new FilterRegistrationBean<>(); registrationBean.setFilter(new MyFilter()); registrationBean.addUrlPatterns("/*"); registrationBean.setName("myFilter"); registrationBean.setOrder(1); return registrationBean; } } ``` 这里使用Java Config的方式声明Filter过滤器。在`myFilter()`方法,创建了一个FilterRegistrationBean对象,并设置其Filter、URL模式、名称和执行顺序。最后返回这个对象即可。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值