一.概述
1.初识拦截器
Interceptor 拦截器类似前面学过的过滤器,是可以在action执行前后执行的代码,是我们做Web开发经常用到的技术。比如:权限控制、日志等。我们也可以将多个Interceptor 连在一起组成Interceptor 栈。
Struts2拦截器,每个拦截器类只有一个对象实例,即采用单例模式,所有引用这个拦截器的Action都共享这一拦截器类的实例,因此在拦截器中如果使用类变量,要注意同步的问题。
注意:
(1)拦截器在访问某个Action之前或之后实施拦截。
(2)拦截器是可插拔的,是AOP的一种实现。
(3)对于Action的一些公共处理代码可以被放到拦截器中,比如权限控制、日志等。
(4)拦截器栈
A: 拦截器栈就是将拦截器按一定的顺序连结成一条链。
B: 在访问被拦截的方法或字段时,拦截器链中的拦截器救护按照之前定义的顺序被调用。
(5)实现原理
步骤一: Struts2拦截器的实现原理相对简单,当请求Struts2的Action时,Struts2会查找配置文件。
步骤二: Struts2根据配置实例化对应拦截器对象,然后串成一个列表,最后一个一个的调用列表中的拦截器。
拦截器采用责任链设计模式,在责任链模式中,很多对象由每一个对象对其下家的引用而连接起来形成的一条链,责任链的每一个节点,都可以继续调用下一个节点,也可以阻止流程继续执行。
Struts2工作原理
(1)客户端初始化一个指向Servlet容器的请求。
(2)这个请求经过一系列的过滤器,接着StrutsPrepareAndExecuteFilter被调用,它询问ActionMapper来决定是否要调用这个某个Action。
(3)如果ActionMapper决定要调用某个Action,StrutsPrepareAndExecuteFilter把请求的处理交给ActionProxy。
(4)ActionProxy通过Configuration Manager询问框架的配置文件,找到需要调用的Action类。
(5)ActionProxy创建一个ActionInvocation的实例。
(6)ActionInvocation实例使用命名模式来调用,在调用Action的前后,经过拦截器的调用。
(7)一旦Actio执行完毕,ActionInvocation负责根据struts.xml配置找到相应返回结果,返回结果通常是JSP或者FreeMaker的模板。
2.拦截器和过滤器的区别
(1)拦截器是基于Java反射机制的。
过滤器是基于函数回调的。
(2)拦截器隶属于Struts2框架,只能拦截Action,无法拦截对JSP的请求。
过滤器隶属于Web容器,可以过滤一切请求,包括JSP、Servlet、Action、HTML等。
(3)在Action的生命周期中,拦截器可以被多次调用。
过滤器只能在容器初始化时被调用一次。
二、Interceptor接口
(1)每个拦截器都是实现了com.opensymphony.xwork2.interceptor.Interceptor接口的 Java 类。
public interface Interceptor extends Serializable { void destroy(); void init(); String intercept(ActionInvocation invocation) throws Exception; }
init: 该方法将在拦截器被创建后立即被调用,它在拦截器的生命周期内只被调用一次。
destroy: 该方法将在拦截器被销毁之前被调用,它在拦截器的生命周期内也只被调用一次。
intercept: 每拦截一个请求动作,该方法都会被调用一次。
(2)Struts 会依次调用为某个 Action 而注册的每一个拦截器的interecept 方法。
(3)每次调用interecept方法时, Struts 会传递一个ActionInvocation 接口的实例。
(4)ActionInvocation: 代表一个给定动作的执行状态, 拦截器可以从该类的对象里获得与该动作相关联的 Action 对象和 Result 对象. 在完成拦截器自己的任务之后, 拦截器将调用 ActionInvocation 对象的 invoke 方法前进到 Action 处理流程的下一个环节。
(5)还可以调用 ActionInvocation 对象的 addPreResultListener 方法给 ActionInvocation 对象“挂”上一个或多个 PreResultListener 监听器. 该监听器对象可以在动作执行完毕之后, 开始执行动作结果之前做些事情。
(6)AbstractInterceptor 类实现了 Interceptor 接口. 并为 init, destroy 提供了一个空白的实现。
(7)Struts2自带的拦截器
三、自定义拦截器
1.编写拦截器
需要实现Interceptor 接口,实现接口中的三个方法,但是Interceptor 接口有很多实现类,编写最简单的方式就是继承AbstractInterceptor类。
public class TestInterceptor extends AbstractInterceptor{ @Override
public String intercept(ActionInvocation invocation) throws Exception{ System.out.println("拦截器执行1....");
invocation.invoke();
System.out.println("拦截器执行2....");
return null;
}
}
2.配置拦截器
需要在struts.xml中进行拦截器的配置。
<package name="user" extends="struts-default"> <!-- 配置拦截器 -->
<interceptors>
<interceptor name="testInterceptor" class="com.kiwi.action.TestInterceptor"></interceptor>
</interceptors> <action name="interceptAction" class="com.kiwi.action.InterceptorTestAction" method="save"> <!-- 引用拦截器 -->
<interceptor-ref name="testInterceptor"/>
<!-- 引用默认的拦截器栈 -->
<interceptor-ref name="defaultStack"/> <result name="success">/success.jsp</result>
</action> </package>
注意:
(1)如果引用了自定义拦截器时,默认拦截器将不会起作用。
(2)默认拦截器在struts-default.xml中,配置了默认拦截器,当配置了默认拦截器以后,如果不引用拦截器,那么默认拦截器将起作用。
(3)当action的拦截器比较多时,可以将多个拦截器放入一个拦截器栈中。
<package name="user" extends="struts-default"> <!-- 配置拦截器 -->
<interceptors> <interceptor name="testInterceptor" class="com.kiwi.action.TestInterceptor"></interceptor> <!-- 配置一个拦截器栈,可以包含多个拦截器的引用 -->
<interceptor-stack name="myStack">
<interceptor-ref name="testInterceptor"/>
<interceptor-ref name="defaultStack"/>
</interceptor-stack> </interceptors> <action name="interceptAction" class="com.kiwi.action.InterceptorTestAction" method="save"> <!-- 引用拦截器栈 -->
<interceptor-ref name="myStack"/> <result name="success">/success.jsp</result> </action> </package>