过滤器(Filter)和拦截器(Interceptor)的区别

本文深入解析了Filter与Interceptor的工作原理及应用场景,对比了两者在实现机制、依赖环境、调用次数等方面的差异,以及如何在实际项目中创建和配置这两种组件。

过滤器(Filter)

依赖于servlet容器。在实现上基于函数回调,可以对几乎所有请求进行过滤,但是缺点是一个过滤器实例只能在容器初始化时调用一次。使用过滤器的目的是用来做一些过滤操作,获取我们想要获取的数据,比如:在过滤器中修改字符编码;在过滤器中修改HttpServletRequest的一些参数,包括:过滤低俗文字、危险字符等

Filter用处
  • 在HttpServletRequest到达Servlet之前,拦截客户的HttpServletRequest。
  • 根据需要检查HttpServletRequest,也可以修改HttpServletRequest头和数据。
  • 在HttpServletResponse到达客户端之前,拦截HttpServletResponse。
  • 根据需要检查HttpServletResponse,也可以修改HttpServletResponse头和数据。
Filter 常用场景
  • 用户授权的Filter:Filter负责检查用户请求,根据请求过滤用户非法请求。

  • 日志Filter:详细记录某些特殊的用户请求。

  • 负责解码的Filter:包括对非标准编码的请求解码。

  • 能改变XML内容的XSLT Filter等。

  • Filter可以负责拦截多个请求或响应;一个请求或响应也可以被多个Filter拦截。

    创建Filter只需两个步骤
  • 创建Filter处理类

  • web.xml文件中配置Filter或Config配置类

    创建Filter必须实现javax.servlet.Filter接口,在该接口中定义了如下三个方法
  • void init(FilterConfig config):用于完成Filter的初始化。

  • void destory():用于Filter销毁前,完成某些资源的回收。

  • void doFilter(ServletRequest request,ServletResponse response,FilterChain chain):实现过滤功能,该方法就是对每个请求及响应增加的额外处理。该方法可以实现对用户请求进行预处理(ServletRequest request),也可实现对服务器响应进行后处理(ServletResponse response)—它们的分界线为是否调用了chain.doFilter(),执行该方法之前,即对用户请求进行预处理;执行该方法之后,即对服务器响应进行后处理

拦截器(Interceptor)

依赖于web框架,在SpringMVC中就是依赖于SpringMVC框架。在实现上基于Java的反射机制,属于面向切面编程(AOP)的一种运用。由于拦截器是基于web框架的调用,因此可以使用Spring的依赖注入(DI)进行一些业务操作,同时一个拦截器实例在一个controller生命周期之内可以多次调用。但是缺点是只能对controller请求进行拦截,对其他的一些比如直接访问静态资源的请求则没办法进行拦截处理

拦截器,在AOP(Aspect-Oriented Programming)中用于在某个方法或字段被访问之前,进行拦截,然后在之前或之后加入某些操作。拦截是AOP的一种实现策略。
在WebWork的中文文档的解释为—拦截器是动态拦截Action调用的对象。它提供了一种机制可以使开发者可以定义在一个Action执行的前后执行的代码,也可以在一个Action执行前阻止其执行。同时也提供了一种可以提取Action中可重用的部分的方式。

拦截器将Action共用的行为独立出来,在Action执行前后执行。这也就是我们所说的AOP,它是分散关注的编程方法,它将通用需求功能从不相关类之中分离出来;同时,能够共享一个行为,一旦行为发生变化,不必修改很多类,只要修改这个行为就可以。

拦截器将很多功能从我们的Action中独立出来,大量减少了我们Action的代码,独立出来的行为就有很好的重用性。

当你提交对Action的请求时,ServletDispatcher会根据你的请求,去调度并执行相应的Action。在Action执行之前,调用被Interceptor截取,Interceptor在Action执行前后执行

  • 创建Interceptor必须实现com.opensymphony.xwork2.interceptor.Interceptor接口,该接口定义了如下三个方法

void init():在该拦截器被实例化之后,在该拦截器执行拦截之前,系统将回调该方法。对于每个拦截器而言,其init()方法只执行一次。因此,该方法的方法体主要用于初始化资源。
void destory():该方法与init()方法对应。在拦截器实例被销毁之前,系统将回调该拦截器的destory方法,该方法用于销毁在init方法里打开的资源。
String intercept(ActionInvocation invocation):该方法是用户需要实现的拦截动作。就像Action的execute方法一样。intercept方法会返回一个字符串作为逻辑视图。如果该方法直接返回了一个字符串,系统会将跳转到该逻辑视图对应的实际视图资源,不会调用被拦截的Action。该方法的ActionInvocation参数包含了被拦截的Action的引用,可以通过调用该参数的invoke方法,将控制权转给下一个拦截器,或者转给Action的execute方法(如果该拦截器后没有其他拦截器,则直接执行Action的execute方法)

过滤器(Filter)和拦截器(Interceptor)的区别

  • 拦截器是基于java的反射机制的,而过滤器是基于函数回调。

  • 拦截器不依赖与servlet容器,过滤器依赖与servlet容器。

  • 拦截器只能对action请求起作用,而过滤器则可以对几乎所有的请求起作用

  • 拦截器可以访问action上下文、值栈里的对象,而过滤器不能访问

  • 在action的生命周期中,拦截器可以多次被调用,而过滤器只能在容器初始化时被调用一次

  • 拦截器可以获取IOC容器中的各个bean,而过滤器就不行,这点很重要,在拦截器里注入一个service,可以调用业务逻辑

Filter和Interceptor的执行顺序

过滤前-拦截前-action执行-拦截后-过滤后

Spring中,过滤器Filter)、拦截器InterceptorAOP(面向切面编程)既有区别又有联系。 ### 区别 - **实现机制**:拦截器基于Java的反射机制,使用代理模式;而过滤器基于函数回调。拦截器不依赖servlet容器,过滤器依赖于servlet容器[^2]。 - **作用范围**:拦截器只能对action起作用,而过滤器可以对几乎所有的请求起作用,能够保护资源[^2]。 - **访问权限**:拦截器可以访问action上下文堆栈里面的对象,过滤器则不能[^2]。 - **执行顺序**:执行顺序为过滤前 - 拦截前 - Action处理 - 拦截后 - 过滤后[^2]。 ### 联系 它们都可以对程序的流程进行拦截处理,起到增强功能控制访问的作用。例如在Spring Boot中,都可以对请求进行一定的前置或后置处理。而且它们都可以用于实现一些通用的功能,如日志记录、权限验证等。 ### 代码示例 以下是简单的过滤器拦截器AOP的代码示例: #### 过滤器示例 ```java import javax.servlet.*; import java.io.IOException; public class MyFilter implements Filter { @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { // 过滤前处理 System.out.println("Filter: Before processing"); chain.doFilter(request, response); // 过滤后处理 System.out.println("Filter: After processing"); } } ``` #### 拦截器示例 ```java import org.springframework.web.servlet.HandlerInterceptor; import org.springframework.web.servlet.ModelAndView; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; public class MyInterceptor implements HandlerInterceptor { @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { // 拦截前处理 System.out.println("Interceptor: Before handler execution"); return true; } @Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { // 拦截后处理 System.out.println("Interceptor: After handler execution"); } } ``` #### AOP示例 ```java import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.After; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut; import org.springframework.stereotype.Component; @Aspect @Component public class MyAspect { @Pointcut("execution(* com.example.service.*.*(..))") public void serviceMethods() {} @Before("serviceMethods()") public void beforeServiceMethod(JoinPoint joinPoint) { // 增强处理(前置通知) System.out.println("AOP: Before service method execution"); } @After("serviceMethods()") public void afterServiceMethod(JoinPoint joinPoint) { // 增强处理(后置通知) System.out.println("AOP: After service method execution"); } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值