Tomcat7下Filter执行顺序小验证

本文解析了两个Filter在不同情况下的执行顺序,包括初始化方法、同一映射URL的处理方法及不同映射URL的情况,并介绍了Tomcat7中Filter的实现。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

定义2个Filter,filter1过滤.htm和.json结尾的url,filter2过滤.htm和.xml结尾的url。
1、init()方法执行顺序:先执行filter2的init()方法,后执行filter1的init()方法。
因为Filter是责任链模式的,上一级对象包含下一级对象的指针,所以创建顺序在web.xml中是从后往前的。
2、同一mapping url的doFilter()执行顺序,如*.htm:先执行filter1的doFilter()方法,后执行filter2的doFilter()方法,会进行链式调用。在web.xml中,是按照从前往后的顺序执行。
3、当全部Filter执行完成,且没有return时,会调用url映射的servlet的service()方法。service()方法执行完后,会逐层放回Filter的doFilter()方法,有点像spring的aop。
4、不同mapping url的doFilter()执行顺序,如*.json:只会调用filter1的doFilter()方法。mapping url只会执行对应Filter的doFilter()方法。
5、Tomcat7的Filter实现类为ApplicationFilterChain,创建代码如下:

ApplicationFilterFactory factory =
ApplicationFilterFactory.getInstance();
ApplicationFilterChain filterChain =
factory.createFilterChain(request, wrapper, servlet);



<filter>
<filter-name>filter1</filter-name>
<filter-class>filter.Filter1</filter-class>
</filter>
<filter-mapping>
<filter-name>filter1</filter-name>
<url-pattern>*.htm</url-pattern>
</filter-mapping>
<filter-mapping>
<filter-name>filter1</filter-name>
<url-pattern>*.json</url-pattern>
</filter-mapping>

<filter>
<filter-name>filter2</filter-name>
<filter-class>filter.Filter1</filter-class>
</filter>
<filter-mapping>
<filter-name>filter2</filter-name>
<url-pattern>*.htm</url-pattern>
</filter-mapping>
<filter-mapping>
<filter-name>filter2</filter-name>
<url-pattern>*.xml</url-pattern>
</filter-mapping>
<think>好的,我现在需要回答用户关于如何在@WebFilter中设置过滤器执行顺序的问题。首先,我要回忆一下Java Web中过滤器的相关知识,特别是使用注解配置的情况。 用户之前询问了拦截器和过滤器的区别和应用场景,现在转向了具体的过滤器顺序设置。我知道在传统的web.xml配置中,过滤器的执行顺序是由<filter-mapping>的声明顺序决定的,但用户问的是使用@WebFilter注解的情况,这需要不同的处理方式。 首先,我需要确认@WebFilter是否支持直接指定执行顺序。根据记忆,@WebFilter注解本身并没有提供显式的顺序属性,比如order或priority之类的参数。这意味着仅仅通过注解可能无法直接控制顺序,这可能导致依赖容器(如Tomcat)的实现细节,而不同容器可能有不同的处理方式,这可能不可靠。 接下来,我需要思考如何间接控制过滤器的执行顺序。可能的解决方案包括: 1. 类名排序:某些容器可能按类名的字母顺序加载过滤器。例如,FilterA会在FilterB之前执行,如果类名按字母排序的话。但这种方法不可靠,且不直观,不推荐在实际开发中使用。 2. 使用web.xml配合:虽然用户使用了注解,但可能可以结合web.xml中的<filter-mapping>来明确指定顺序。在web.xml中声明过滤器的顺序会覆盖注解的配置,因此可以通过这种方式控制顺序。但用户可能希望完全使用注解,所以需要提到这种方法的可行性。 3. 使用Spring的Order注解或Ordered接口:如果项目中使用的是Spring Boot,可能可以通过@Order注解或实现Ordered接口来指定过滤器的顺序。不过需要注意的是,原生的@WebFilter可能不与Spring的机制直接兼容,可能需要将过滤器注册为Spring的Bean,并配合FilterRegistrationBean来调整顺序。 4. 使用FilterRegistrationBean:在Spring Boot中,可以通过编程的方式注册过滤器,并使用FilterRegistrationBean来设置顺序。这种方法比较可靠,并且能够明确控制顺序,应该是推荐的做法。 然后,我需要验证这些方法的正确性。例如,检查Spring Boot文档,确认FilterRegistrationBean的使用方式。根据记忆,FilterRegistrationBean允许设置order属性,通过setOrder方法,值越小优先级越高。这确实是一个有效的方法。 同时,需要提到原生的@WebFilter在纯Servlet容器中的局限性,即无法保证顺序,可能依赖于类加载顺序,而不同容器的行为不同,导致不可预测的结果。因此,在需要精确控制顺序时,最好使用Spring Boot的FilterRegistrationBean方式。 然后,用户可能需要具体的代码示例。例如,如何通过FilterRegistrationBean注册过滤器并设置顺序。需要给出一个例子,展示在Spring Boot配置类中定义多个FilterRegistrationBean,并设置它们的order值。 另外,需要提醒用户,当使用@WebFilterFilterRegistrationBean同时注册同一个过滤器时,可能会导致重复注册的问题,因此应该选择其中一种方式,通常推荐使用FilterRegistrationBean来更好地控制。 最后,总结一下:在原生Servlet中,@WebFilter顺序不可靠,建议结合web.xml或使用Spring Boot的FilterRegistrationBean来明确指定顺序,后者更为推荐,特别是在Spring项目中。</think>在Servlet规范中,使用`@WebFilter`注解的过滤器默认没有显式的顺序控制机制,但可通过以下方案实现顺序控制: --- ### 一、原生Servlet方案(不推荐) ```java // 通过类名字母排序(依赖容器实现) @WebFilter("/*") public class AFilter implements Filter { /* 第一个执行 */ } @WebFilter("/*") public class BFilter implements Filter { /* 第二个执行 */ } ``` **缺点**:不同容器(Tomcat/Jetty等)实现可能不同,不可靠 --- ### 二、Spring Boot推荐方案 #### 方案1:使用FilterRegistrationBean(最佳实践) ```java @Configuration public class FilterConfig { // 低order值先执行 @Bean public FilterRegistrationBean<AFilter> aFilter() { FilterRegistrationBean<AFilter> bean = new FilterRegistrationBean<>(); bean.setFilter(new AFilter()); bean.addUrlPatterns("/*"); bean.setOrder(1); // 顺序值越小优先级越高 return bean; } @Bean public FilterRegistrationBean<BFilter> bFilter() { FilterRegistrationBean<BFilter> bean = new FilterRegistrationBean<>(); bean.setFilter(new BFilter()); bean.addUrlPatterns("/*"); bean.setOrder(2); return bean; } } ``` #### 方案2:结合@Order注解(需配合@Component) ```java @Component @Order(1) @WebFilter("/*") public class AFilter implements Filter { ... } @Component @Order(2) @WebFilter("/*") public class BFilter implements Filter { ... } ``` **注意**:需要添加`@Component`注解并确保包扫描生效 --- ### 三、顺序验证方法 ```java // 在过滤器中打印日志验证执行顺序 public void doFilter(...) { System.out.println("AFilter 开始执行"); chain.doFilter(request, response); System.out.println("AFilter 结束执行"); } ``` --- ### 四、执行顺序规则 1. **正序执行**:order值越小越早执行`doFilter()` 2. **逆序完成**:order值越小越晚执行`chain.doFilter()`之后的代码 3. **典型顺序值参考**: - 安全过滤:Ordered.HIGHEST_PRECEDENCE(值最小) - 日志记录:Ordered.LOWEST_PRECEDENCE(值最大) --- ### 五、注意事项 1. 避免同时使用`@WebFilter`和`FilterRegistrationBean`注册同一过滤器 2. 静态资源过滤器需明确指定`DispatcherType`: ```java bean.setDispatcherTypes(DispatcherType.REQUEST, DispatcherType.FORWARD); ``` 3. 异步请求处理需启用`asyncSupported`: ```java @WebFilter(urlPatterns = "/*", asyncSupported = true) ``` 建议在Spring Boot项目中优先使用`FilterRegistrationBean`方案,既能精确控制顺序,又能兼容各种Servlet容器。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值