对过滤器Filter的思考

本文通过两个模拟程序探讨了JSP中过滤器的工作原理及其实现方式,包括使用集合管理和责任链模式两种方法。

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

用过JSP编程的人都应该知道有个叫filter的东西。 过滤器非常有用,经常用于对字符编程的过滤,也可用于对请求进行统一验证等等。它的过程不难理解,感觉无非是拦截了请求,过滤后再传给其它filter或servlet又或jsp页面。具体理解起来,还挺多东西的。

 刚看了《J2EE核心模式》(j2ee core pattern)过滤器,觉得有些东西需要弄明白,就写了个模拟程序。

import java.util.*;

class Test {
    
public static void main(String[] args) {
        Filter[] filters 
= new Filter[2];
        filters[
0= new AddGoodFilter();
        filters[
1= new AddBadFilter();
        Target target 
= new Target();
        FilterManager fm 
= new FilterManager();
        fm.addFilter(filters[
0]);
        fm.addFilter(filters[
1]);
        fm.setTarget(target);
        StringBuffer sb 
= new StringBuffer();
        sb.append(
"After filter : ");
        
new Client(fm).request(sb);
    }


}


class Client {
    
private FilterManager manager;

    
public Client(FilterManager manager) {
        
this.manager = manager;
    }


    
public void request(StringBuffer sb) {
        System.out.println(manager.request(sb));
    }

}


class FilterManager {
    
private Target target;

    
private ArrayList<Filter> filters = new ArrayList<Filter>();

    
public void setTarget(Target target) {
        
this.target = target;
    }


    
public void addFilter(Filter filter) {
        filters.add(filter);
    }


    
public StringBuffer request(StringBuffer sb) {
        sb.append(
" filter begin: ");
        
for (Iterator<Filter> it = filters.iterator(); it.hasNext();) {
            it.next().doFilter(sb);
        }

        sb.append(
" toTarget: ");
        
return target.request(sb);
    }


}


interface Filter {
    
public void doFilter(StringBuffer sb);
}


class AddGoodFilter implements Filter {
    
public void doFilter(StringBuffer sb) {
        sb.append(
"-good");
    }

}


class AddBadFilter implements Filter {
    
public void doFilter(StringBuffer sb) {
        sb.append(
"-bad");
    }

}


class Target {
    
public StringBuffer request(StringBuffer sb) {
        sb.append(
"-target");
        
return sb;
    }

}

从执行结果可以看到,目标类(Target)只向请求里添加 "-target" 字符串,结果因为经过了过滤器,返回就不只是"-target" 字符串。

以上的实现参照了《J2EE核心模式》书中的示例代码。写完了,运行模拟了一下,觉得效果有点像,虽然离真正标准的过滤器还差得远。所以我决定去查看下JSP的源码,结果失望而归。因为Filter接口的代码只有这么几行。

public interface Filter {

    
public void init(FilterConfig filterConfig) throws ServletException;

    
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException,ServletException;

    
public void destroy();
}

从这几行代码能看出什么来吗?我看不出名堂来,只知道接口是定义操作,真正的具体的实现工作让容器(比如Tomcat)做了。那可能要去看Tomcat的源码才能完全了解 Filter。算了,不理它。

听说过滤器的实现是使用责任链模式来实现的。于是我再写一个模拟程序,大概用到了责任链模式(因为不熟,只能写个样子)。


import java.util.*;
//测试
class Test {
    
public static void main(String[] args) {
        Filter[] filters 
= new Filter[2];
        filters[
0= new AddGoodFilter();
        filters[
1= new AddBadFilter();
        Target target 
= new Target();
        FilterManager fm 
= new FilterManager();
        fm.setFirstFilter(filters[
1]);
        filters[
1].setNextFilter(filters[0]);
        fm.setTarget(target);
        StringBuffer sb 
= new StringBuffer();
        sb.append(
"After filter : ");
        
new Client(fm).request(sb);
    }


}

//客户请求
class Client {
    
private FilterManager manager;

    
public Client(FilterManager manager) {
        
this.manager = manager;
    }


    
public void request(StringBuffer sb) {
        System.out.println(manager.request(sb));
    }

}

//过滤器管理类
class FilterManager {
    
private Target target;

    
private Filter filter;

    
public void setTarget(Target target) {
        
this.target = target;
    }


    
public void setFirstFilter(Filter filter) {
        
this.filter=filter;
    }


    
public StringBuffer request(StringBuffer sb) {
        sb.append(
" filter begin: ");
        filter.doFilter(sb);
        sb.append(
" toTarget: ");
        
return target.request(sb);
    }


}

//过滤器接口
interface Filter {
    
public void doFilter(StringBuffer sb);
    
public void setNextFilter(Filter filter);
}

//
class AddGoodFilter implements Filter {
    
private Filter filter;
    
public void setNextFilter(Filter filter)
    
{
        
this.filter=filter;
    }

    
public void doFilter(StringBuffer sb) {
        sb.append(
"-good");
        
if(filter!=null)filter.doFilter(sb);
    }

}

//
class AddBadFilter implements Filter {
    
private Filter filter;
    
public void setNextFilter(Filter filter)
    
{
        
this.filter=filter;
    }

    
public void doFilter(StringBuffer sb) {
        sb.append(
"-bad");
        
if(filter!=null)filter.doFilter(sb);
    }

}

//请求目标类
class Target {
    
public StringBuffer request(StringBuffer sb) {
        sb.append(
"-target");
        
return sb;
    }

}

目前对过滤器的理解就只能到这里了。以后有空再去学习学习下。

 

<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容器。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值