gistfile1.txt 拦截器(interceptor)是Struts2最强大的特性之一,也可以说是struts2的核心,拦截器可以让你在Action和result被执行之前或之后进行一些处理。同时,拦截器也可以让你将通用的代
拦截器(interceptor)是Struts2最强大的特性之一,也可以说是struts2的核心,拦截器可以让你在Action 和result被执行之前或之后进行一些处理。同时,拦截器也可以让你将通用的代码模块化并作为可重用的类。Struts2中 的很多特性都是由拦截器来完成的。拦截是AOP(Aspect Objected Programing:面向切面编程)的一种实现策略。 在Webwork的中文文档的解释为:拦截器是动态拦截Action调用的对象。它提供了一种机制可以使开发者可以定义在一个 action执行的前后执行的代码,也可以在一个action执行前阻止其执行。同时也是提供了一种可以提取action中可重用 的部分的方式。谈到拦截器,还有一个词大家应该知道——拦截器链(Interceptor Chain,在Struts 2中称为 拦截器栈Interceptor Stack)。拦截器链就是将拦截器按一定的顺序联结成一条链。在访问被拦截的方法或字段时, 拦截器链中的拦截器就会按其之前定义的顺序被调用。 一.拦截器的实现原理: 大部分时候,拦截器方法都是通过代理的方式来调用的。Struts 2的拦截器实现相对简单。当请求到达Struts 2的 ServletDispatcher时,Struts 2会查找配置文件,并根据其配置实例化相对的拦截器对象,然后串成一个列表(list), 最后一个一个地调用列表中的拦截器。事实上,我们之所以能够如此灵活地使用拦截器,完全归功于“动态代理”的使用。 动态代理是代理对象根据客户的需求做出不同的处理。对于客户来说,只要知道一个代理对象就行了。那Struts2中, 拦截器是如何通过动态代理被调用的呢? 当Action请求到来的时候,会由系统的代理生成一个Action的代理对象,由这 个代理对象调用Action的execute()或指定的方法,并在 struts.xml中查找与该Action对应的拦截器。如果有对应的 拦截器,就在Action的方法执行前(后)调用这些拦截器;如果没有对应的拦截 器则执行Action的方法。其中系统对于 拦截器的调用,是通过ActionInvocation来实现的,会使用invoke()方法调用下个拦截器,若没有则执行action,如下: public String intercept(ActionInvocation invocation) throws Exception { String result = null; before(invocation); // 调用下一个拦截器,如果拦截器不存在,则执行Action result = invocation.invoke(); after(invocation, result); return result; } Struts2的拦截器结构的设计,实际上是一个典型的责任链模式的应用。首先将整个执行划分成若干相同类型的元素,每个 元素具备不同的逻辑责任,并将他们纳入到一个链式的数据结构中(我们可以把堆栈结构也看作是一个递归的链式结构), 而每个元素又有责任负责链式结构中下一个元素的执行调用。 拦截器中的返回值result就是action中返回的字符串,也是struts.xml中对应action的执行。 我们可以以invocation.invoke()为界,将拦截器中的代码分成2个部分: 1)before拦截器:在invocation.invoke()之前的代码,将会在invocation.invoke()代码执行之前执行,即在Action之前被依次顺序执行, 2)after拦截器:在invocation.invoke()之后的代码,,将会在invocation.invoke()代码执行之后执行,即在Action之后被逆序执行。 3)PreResultListener拦截器: 有的时候,before拦截和after拦截对我们来说是不够的,因为我们需要在Action执行完之后, 但是还没有回到视图层之前,做一些事情。 Struts2同样支持这样的拦截,这种拦截方式,是通过在拦截器中注册一个PreResultListener的接口来实现的。 下面是接口示例: public interface PreResultListener { void beforeResult(ActionInvocation invocation, String resultCode); } 下面是拦截器注册示例: public String intercept(ActionInvocation invocation) throws Exception { invocation.addPreResultListener(new PreResultListener() { public void beforeResult(ActionInvocation invocation, String resultCode) { /********** 要添加的代码 ***********/ } }); String returnString = invocation.invoke(); return returnString; } 注意:在addPreResultListener里的异常,不会被Struts的框架捕获 由此,我们就可以通过invocation.invoke()作为Action代码真正的拦截点,从而实现AOP。