SptingBoot过滤器Filter的使用方式

时间:2022-11-04 15:56:06


Filter属于Servlet规范,并不是Spring独有的。Filter主要用于拦截请求,做一些业务逻辑操作,然后可以决定请求是否可以继续往下分发,落到其他的Filter或者对应的Servlet。

Filter的工作流程

  1. 进入filter,执行相关业务逻辑

  2. 若判定失败,直接返回,不需要将请求发给Servlet

  3. 若判定通行,进行入下一个filter

    如果全部filter通行,进入Servlet逻辑,Servlet执行完毕之后,又返回Filter,最后在返回给请求方

自定义Filter的使用方式

Spring中过滤器Filter不同使用方式都是通过:FilterRegistrationBean包装filter,最终注册到Servlet容器中。

1. @WebFilter+@ServletComponentScan

在SpringBootApplication上添加@ServletComponentScan注解,在Filter上添加@WebFilter注解。

该方式的缺点:无法设置过滤器之间的优先级。

@WebFilter+@ServletComponentScan方式无法通过@Order注解指定过滤器优先级:优先级使用默认值Ordered.LOWEST_PRECEDENCE(2147483647),相同优先级的情况下,根据名字先后顺序来决定。

使用方法

1. 在SpringBootApplication上使用@ServletComponentScan注解

@ServletComponentScan
@SpringBootApplication
public class SpringbootApplication {

    public static void main(String[] args) {
        SpringApplication.run(SpringbootApplication.class, args);
    }
}

2.在Filter上使用@WebFilter注解

@WebFilter(urlPatterns = {"/test3"})
public class MyFilter extends OncePerRequestFilter {

    @Override
    protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException {
        System.out.println("======= MyFilter =======");
        filterChain.doFilter(request,response);
    }
}

@WebFilter

@WebFilter用于将一个类声明为过滤器。

WebFilter注解的主要属性:

属性名 类型 描述
filterName String 指定过滤器的name属性,(springbean也是用该名称),等价于< filter-name>
urlPatterns String[] 指定一组过滤器的URL匹配模式。等价于< url-pattern>
value String[] 该属性等价于urlPatterns属性,但是两个不应该同时使用
sevletNames String[] 指定过滤器将用于哪些servlet。取值是@WebServlet中的name属性的取值,或者是web.xml中< servlet-name>
dispatcherTypes DispatcherType[] 指定一组过滤器的转发模式。具体取值包括:ASYNC、ERROR、FORWARD、INCLUDE、REQUEST,默认REQUEST
initParams WebInitParam[] 指定一组过滤器初始化参数,等价于< init-param>
asyncSupported boolean 声明过滤器是否支持异步操作模式,等价于< async-supported>标签
description String 过滤器的描述信息,等价于< description>
displayName String 过滤器的显示名,通常配合工具使用,等价于< display-name>

@ServletComponentScan

在SpringBoot项目中,@WebServlet、@WebFilter、@WebListener这三个注解默认是不会被扫描的,一般在SpringBootApplication上添加@ServletComponentScan注解,表示对这三个注解的扫描。

@ServletComponentScan可以实现將Servlet(控制器)、Filter(过滤器)、Listener(监听器)自动注册到Spring容器中,无需其他代码。

  • Servlet:通过@WebServlet注解定义
  • Filter:通过@WebFilter注解定义
  • Listener:通过@WebListener注解定义

常见问题分析

1. 只使用@WebFilte:过滤器不生效

WebFilter属于注解属于Servlet3+,与Spring本身没有什么关系,所以Spring默认是不认识这个注解的。

2. @WebFilter+@Component:配置的过滤条件不生效
同上,Spring是不认识@WebFilter注解的,所以注解配置的任何属性都无意义(例如:指定过滤的url)。
该种方式其实就等同于只加了个@Component注解,此时过滤器能生效,但无过滤条件,会过滤所有url。

3. @WebFilter+@Component+@ServletComponentScan:过滤器会被调用两次

  • 一次:@WebFilter+@ServletComponentScan,被SpringBean管理,过滤器生效,根据@WebFilter配置的属性过滤
  • 二次:@Component,又被SpringBean管理一次,过滤器生效(和上面不是同一个bean),过滤全部url

4. @WebFilter+@Order+@ServletComponentScan:设置过滤器优先级无效

通过@WebFilter+@ServletComponentScan方式注册的,其生成的FilterRegistrationBean并没有检查@Order注解,所以@Order注解不生效。

细节可参考文章:关于@webFilter使用@Order无效问题

2. @Component+@Order

通过在Filter上加@Component和@Order注解,即可被Spring管理,并可指定过滤器的执行顺序。

该方式的缺点:只能过滤所有URL,不能通过配置去过滤指定的 URL。

使用方法

@Order(100)
@Component
public class MyFilter extends OncePerRequestFilter {

    @Override
    protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException {
        System.out.println("======= MyFilter =======");
        filterChain.doFilter(request,response);
    }
}

3. FilterRegistrationBean(推荐)

直接通过配置类定义Filter的FilterRegistrationBean,交给SpringBean容器管理。

该方式既能通过配置去过滤指定的 URL,也能指定过滤器之间的优先级。

使用方法

public class MyFilter extends OncePerRequestFilter {

    @Override
    protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException {
        System.out.println("======= MyFilter =======");
        filterChain.doFilter(request,response);
    }
}
@Configuration
public class FilterConfig {

    @Bean
    public FilterRegistrationBean<MyFilter> filterRegistrationBean() {
        FilterRegistrationBean registration = new FilterRegistrationBean();
        registration.setFilter(myFilter());
        registration.setName("myFilter");
        registration.addUrlPatterns("/test3");
        registration.setOrder(100);
        return registration;
    }

    @Bean
    public MyFilter myFilter() {
        return new MyFilter();
    }
}