Struts2 中拦截器和Action的调用关系

所谓的拦截器就是指实现了Interceptor接口的类,该接口中定义了三个方法:init(),destroy(),intercept()。init()用于在服务器启动的时候,初始化这个拦截器。destroy()用于清理分配给拦截器的资源,intercept()是用来起拦截作用的,这个方法还有一个ActionInvocation类型的参数invocation,并且返回一个字符串。ActionInvocation类中有一个方法,是invoke(),这个方法很重要,它的作用是首先判断一下有没有其他的拦截器了,如果有,则继续进入下一个拦截器,如果没有那么就进入Action中执行业务逻辑。intercept()方法的执行过程也很特殊,在invoke()函数之前的代码先执行,执行到invoke()时,判断一下,进入下一个拦截器,或是进行Action的业务处理,等到执行完所有请求转发的Action时,会再回到intercept()方法,继续执行invoke()后面的内容。

注意上面红颜色的部分,为什么是执行完所有的请求转发的Action呢?这是因为执行完当前的Action,可能会请求转发或是重定向到另一个Action中或是结果页面,若是重定向的话,那么就不是在一个请求过程中了,就会重新发一次请求,请求另一个Action,因为一个拦截器只会对一个特定的Action起作用,所以重定向,就完全是另一次请求过程了。然而在拦截器中,若是重定向到另一个Action的话,那么当前的Action就算完成了它的业务逻辑,就会返回执行invoke()方法之后的内容,然后,再进行另一次请求过程。如果是请求转发的话,那么Action之间是在一个请求过程中,等到所有的请求转发的Action执行完之后,才会去执行invoke()后面的内容。

下面,举两个例子对比一下:

有一个jsp页面,两个Action:Action1和Action2

第一次:Action1和Action2之间是请求转发关系

第二次:Action1和Action2之间是重定向关系

Login.jsp:
    
<form action="/struts2/test/action1" method="post">
    	姓名:<input type="text" name="username"/><br/>
    	密码:<input type="password" name="password"/><br/>
    	<input type="submit" value="提交"/>
</form>
拦截器:MyInterceptor.java:

package com.suo.interceptor;

import com.opensymphony.xwork2.ActionInvocation;
import com.opensymphony.xwork2.interceptor.Interceptor;

public class MyInterceptor implements Interceptor {

	@Override
	public void destroy() {
		// TODO Auto-generated method stub
		System.out.println("destroy invoke !");
	}

	@Override
	public void init() {
		// TODO Auto-generated method stub
		System.out.println("init invoke !");
	}

	@Override
	public String intercept(ActionInvocation invocation) throws Exception {
		System.out.println("before invoke !");
		
		String result=invocation.invoke();
		
		System.out.println("after invoke !");
		
		return result;
	}

}
Action1.java:

package com.suo.actions;

import com.opensymphony.xwork2.ActionSupport;

public class Action1 extends ActionSupport {
	
	private String username;
	private String password;
	
	public String getUsername() {
		return username;
	}
	public void setUsername(String username) {
		this.username = username;
	}
	public String getPassword() {
		return password;
	}
	public void setPassword(String password) {
		this.password = password;
	}
	
	public String execute()
	{
		System.out.println("action1 invoke !");
		return SUCCESS;
	}
}
Action2.java:

package com.suo.actions;

import com.opensymphony.xwork2.ActionSupport;

public class Action2 extends ActionSupport {
	
	private String username;
	private String password;
	
	public String getUsername() {
		return username;
	}
	public void setUsername(String username) {
		this.username = username;
	}
	public String getPassword() {
		return password;
	}
	public void setPassword(String password) {
		this.password = password;
	}

	public String execute()
	{
		
		System.out.println("action2 invoke !");	
		return SUCCESS;
	}
}

在struts.xml中配置拦截器:
<interceptors>
	<interceptor name="myInterceptor" class="com.suo.interceptor.MyInterceptor"></interceptor>
</interceptors>
在action中,引用这个拦截器,注意,这里Action之间是请求转发的关系:
<action name="action1" class="com.suo.actions.Action1">
	 <result name="success" type="chain"><!--注意这里是请求转发类型-->
	 	<param name="actionName">action2</param>
	 </result>
	 <interceptor-ref name="myInterceptor"></interceptor-ref><!--这里是引用自定义的拦截器-->
	 <interceptor-ref name="defaultStack"></interceptor-ref>
<!--注意,每一个action都有一个默认的拦截器,如果指定了自定义的拦截器,那么默认的拦截器就失去作用了,所以这里要再加上默认的拦截器-->
</action>
	 	
<action name="action2" class="com.suo.actions.Action2">
	 <result name="success">/WEB-INF/result/action.jsp</result>
</action>
启动服务器,可以看到输出的结果是:
before invoke !
action1 invoke !
action2 invoke !
after invoke !
若是将Action间的类型换成redirectAction,那么输出的结果是:
before invoke !
action1 invoke !
after invoke !
action2 invoke !
补充:还可以给拦截器传递参数,进行初始化的工作。如下:
<interceptors>
	 <interceptor name="myInterceptor" class="com.suo.interceptor.MyInterceptor">
	 	<param name="suo">piao</param><!--传递一个参数-->
	 </interceptor>
</interceptors>
然后在拦截器中定义该参数的get/set方法,即可:
注意,这个参数的赋值,要早于init()方法的调用,因为初始化参数的作用,就是想在初始化的时候用到这个参数,所以要早于init()方法。

  • 1
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值