`
xiaotao.2010
  • 浏览: 211971 次
  • 性别: Icon_minigender_1
  • 来自: 成都
社区版块
存档分类
最新评论

Interceptor的初步认识

阅读更多

一个名称 :

AOP(Aspect oriented programming) 面向切面编程

听着这个很神秘,其实不然 简单点说他就是个Interceptor (拦截器)


前两天刚刚学完了Filter(过滤器)被过滤器中的chain.doFilter() 一直搞的晕晕呼呼的,有人说执行了doFilter后返回到Action去,又有人说执行玩doFilter后返回下一个Filter中去,两个观点一直僵持不下。


今天了解了下Interceptor 其中的原理豁然开朗,Interceptor中也有类似doFilter的方法 叫做Invocation.invoke()方法,工作原来和doFilter很相似,暂时先把他们连个执行的原理归为一类。

先上一段代码,来解释Invocation.invoke()方法。

//Interceptor 中的代码
//部分来嘛省略,页面传递值的代码省略。。。

public String intercept(ActionInvocation invocation) throws Exception {
		System.out.println("Now in the Interceptor!");
		System.out.println("Interceptor startup before!");
		String s = invocation.invoke();
		System.out.println("Interceptor startup after!");
		String name = (String)invocation.getInvocationContext().getValueStack().findValue("name");
		System.out.println("In the Interceptor \t name= "+name);
		return s;
	}

 

//Action中的代码
public class TestAction extends ActionSupport{
	public String name,password;
	@Override
	public String execute() throws Exception {
		System.out.println("name:"+name+"\tpassword:"+password);
		return super.execute();
	}
}
  



来看看结果是什么,红色的标注是Action中执行的结果

==============================================================================
//Tomcat中的结果
Now in the Interceptor!

Interceptor startup before!

//突然发现Invocation.invoke()方法成为程序的分割线,程序跳入Action中了
//这一段是Action 中输出的值

name:1231123 password:14213


//好了,在此程序又回到了 interceptor

Interceptor startup after!
In the Interceptor name= 1231123

==============================================================================

好了,到此为止可能还是不明确这个Invocation.invoke()方法执行完成以后是去了Action还是去了哪里。但到此能确定的一点是,程序到Invocation.invoke()这个方法的时候去执行了另一个程序,等到另一个程序执行完成后又回到了这个interceptor。

再上一段代码。

在这做一点说明,以上的代码都不动,只是在FirstInterceptor后面在加上一个Interceptor名字叫做SecondInterceptor。
好了上代码

public class SecondInterceptor implements Interceptor{
	public void destroy() {
	
	}

	public void init() {
	
	}

	public String intercept(ActionInvocation invocation) throws Exception {
		System.out.println("**************************************");
		System.out.println("Now in the SecondInterceptor");
		System.out.println("SecondInterceptor startup before!");
		String s = invocation.invoke();
		System.out.println("SecondInterceptor startup after!");
		String password = (String)invocation.getInvocationContext().getValueStack().findString("password");
		System.out.println("SecondInterceptor ----  password:"+password);
		System.out.println("**************************************");
		return s;
	}
}



好了,我们再来看看结果。同样红色地方是Action输出的结果


==============================================================================


Now in the Interceptor!
Interceptor startup before!

**************************************
Now in the SecondInterceptor
SecondInterceptor startup before!

name:1231123 password:14213


SecondInterceptor startup after!
SecondInterceptor ----  password:14213
**************************************

Interceptor startup after!
In the Interceptor name= 1231123
==============================================================================

结果很有意思,这个不能画图说明问题,很头痛,现在蛮喜欢画图的,要是能画图问题能说明的很清楚。
用文字图来表示吧,尽量清楚点

(忍不下这个编辑器了,每次调好了又变乱了)
                                          request
                                                 |
   进入Action内部此处略去            |
                                    FirstInterceptor
   FirstInterceptor调用               |            输出:
   ActionInvocation中的              |               Now in the Interceptor!
   invoke()方法                           |               Interceptor startup before!
                                     SecondInterceptor
   SecondInterceptor调用          |            输出:
   ActionInvocation中的              |              Now in the SecondInterceptor
   invoke()方法                           |          SecondInterceptor startup before!
                                           Action
     执行execute()方法                |            输出:
                                                |              name:1231123 password:14213 
  (好了在此Interceptor             |
    原路返回)                 SecondInterceptor  
                                                |            输出:
                                                |       SecondInterceptor startup after!
                                                |       SecondInterceptor --assword:14213
                                         FirstInterceptor
                                                |             输出:
                                                |       Interceptor startup after!
                                                |       In the Interceptor name= 1231123
                                           response

   好了到此Interceptor执行完了,中间无关的步骤略去,在此就谈Interceptor工作流程。
从上很明显的可以看出Interceptor调用ActionInvocation中的invoke()方法 的时候,就进入了下一个Interceptor中执行,当下一个Interceptor调用ActionInvocation中的invoke()方法的时候,同样也会向下执行,直到Action之前的Interceptor全部执行完成,Action执行后,数据又会按照Interceptor进来的顺序原路返回。
   就在这一去一会的路上Interceptor能干很多事情,就对这个Action来说Interceptor完全起到了过滤的作用,可以这么说这个Interceptor就是这个Action的过滤器。
好了话题回到了过滤器上了,之前真论的chain.doFilter()有结果,两个的说法都正确,怎么说了,当只有一个Filter的时候执行chain.doFilter()确实将链条传递到了Action手上,Action得以执行,若不写这个chain.doFilter(),好了这个Filter就太自私了,将链条拿在自己手上,不向下传递,直接导致程序无法执行,程序始终处在等待执行的状态。
要是有多个Filter的时候呢,当一个Filter执行chain.doFilter()的时候是讲这个链条交向下一个Filter手中,直到交到Action手中。
   好了Filter和Interceptor的区别就出来了,Filter执行完以后它不会原路返回,而Interceptor调用ActionInvocation中的invoke()方法的时候就会向下执行,最终还会返回这个调用ActionInvocation中的invoke()方法的位置向下执行。
在Filter中一般chain.doFilter()是写在最后确定执行过滤后在执行chain.doFilter(),一去就不再回来了。

好了最后上一段配置文件,方便查看

<struts>
	<package name="user" namespace="/user" extends="struts-default">	
		<interceptors>
			<interceptor name="firstInterceptor" class="com.ibm.interceptor.FirstInterceptor"></interceptor>
			<interceptor name="secondInterceptor" class="com.ibm.interceptor.SecondInterceptor"></interceptor>
		</interceptors>	
		<action name="testAction" class="com.ibm.test.TestAction">
			<interceptor-ref name="defaultStack"></interceptor-ref>
			<interceptor-ref name="firstInterceptor"></interceptor-ref>
			<interceptor-ref name="secondInterceptor"></interceptor-ref>
			<result name="success"> /test/success.jsp</result>
		</action>		
	</package>
</struts>













 

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics