Struts2的拦截器
Struts2拦截器:
一、Struts2的工作原理:
1、Struts2基本工作过程
1) 客户端初始化一个指向Servlet容器(例如Tomcat)的请求
2) 这个请求经过一系列的过滤器(Filter)(这些过滤器中有一个叫做ActionContextCleanUp的可选过滤器,这个过滤器对于Struts2和其他框架的集成很有帮助,例如:SiteMesh Plugin);
3) 接着StrutsPrepareAndExecuteFilter被调用,StrutsPrepareAndExecuteFilter询问ActionMapper来决定这个请求是否需要调用某个Action;
4) 如果ActionMapper决定需要调用某个Action, StrutsPrepareAndExecuteFilter 把请求的处理交给ActionProxy
5) ActionProxy通过Configuration Manager询问框架的配置文件,找到需要调用的Action类;
6) ActionProxy创建一个Action Invocation的实例。
7) Action Invocation实例使用命名模式来调用,在调用Action的过程前后,涉及到相关拦截器(Intercepter)的调用
8) 一旦Action执行完毕,ActionInvocation负责根据struts.xml中的配置找到对应的返回结果。返回结果通常是(但不总是,也可能是另外的一个Action链)一个需要被
示的JSP或者FreeMarker的模版。在表示的过程中可以使用Struts2框架中继承的标签。在这个过程中需要涉及到ActionMapper。
2、拦截器介绍
1)使用拦截器可以实现横切功能并使这些实现相对action甚至Struts2框架保 持独立。
2)可以实现和使用自己所需的特性且不用修改框架的底层代码。
3)使用拦截器可以达到以下目的:
? 在调用Action之前,提供预处理逻辑
? 与Action进行交互,提供执行信息,比如设置请求中的参数
? 在调用Action之后,提供后处理逻辑
? 修改返回的结果,进而修改呈现给用户的
? 捕获异常从而替换可执行的处理过程或返回一个不同结果
4)常见的拦截器和拦截器栈
5)详见struts-default.xml
6)配置一个内置的timer拦截器
获取执行某个Action组件所耗费的时间,以便做一个粗略的性能调试。
在struts.xml中的某个Action组件的配置中添加 interceptor-ref 标签
比如:
action name= helloworld class= demo.web.HelloWorldAction
interceptor-ref name= timer /
result /success.jsp /result
/)action
说明: interceptor-ref 的name属性为拦截器的别名,这里引用的是Struts2的内 置的,具体地在struts-default.xml的struts-default包中定义有。
请求该Action组件观察服务器控制台输出,(以下为参考结果,具体机器可能不一 样)
第一次请求
信息: Executed action [//helloworld!execute] took 356 ms.
第二次请求
信息: Executed action [//helloworld!execute] took 4 ms.
自行分析:为什么第二次所耗费的时间最小?
3、Struts2拦截器有关API
Interceptor接口与AbstractInterceptor
init() 方法用来初始化拦截器
destroy()方法为拦截器提供清理
intercept()方法为拦截器处理业务规则
其中,init()和destroy()仅在Struts2初始化时和框架关闭时分别执行一次, 而intercept()会在每次请求中都会被调用,所以拦截器需要线程安全,尤其 是intercept()方法如果想创建一个自定义拦截器来实现某个特性,则只需要 实现Interceptor接口即可
参考如下:
public class MyTimerInterceptor implements Interceptor {
public void destroy() {
System.out.println( 服务器关闭的时候执行一次destory的方法..... );}
public void init() {
System.out.println( 服务器初始化的时候执行一次init方法.... );}
// 每次请求的时候都执行intercept的方法
public String intercept(ActionInvocation ai) throws Exception {
long beforetime = new Date().getTime();// 得到action被执行之前的时间
String result = ai.invoke(); // 得到后续的拦截器和action的操作
long aftertime = new Date().getTime();// 得到action被执行之后的时间
System.out.println( 执行的action共计执行了: + (aftertime - beforetime));
System.out.println( 返回从action中得到返回值是: +result);
return result;}}
说明:
q 调用invocation.invoke()方法可将请求进一步传递给下一个拦截器去处理
q 否则,请求就不会到达Action组件
struts.xml文件拦截器的配置
package name=“sinter extends= struts-default namespace= /
!-- 声明自定义的拦截器 --
interceptors
interceptor name= MyTimer
class= com.redarmy.intercepter.MyTimerInterceptor /interceptor
/interceptors
action name= login class= com.redarmy.action.LoginAction
!-- 使用自己定义的拦截器 --
interceptor-ref name= MyTimer /interceptor-ref
result name= success /user/reg.jsp /result
result name= input /index.jsp /result
/action
/package
部署运行,查看控制台输出,参考结果如下:
服务器初始化的时候执行一次init方法....
==============login.action
2010-8-16 10:52:56 com.opensymphony.xwork2.util.logging.commons.CommonsLogger
info
信息: Executed action [//login!execute] took 32 ms.
执行的action共计执行了:32
返回从action中得到返回值是:success
如果不需要初始化和清理代码,则可以继承AbstractInterceptor。该类提供了一个默认实现,但是init()和destroy()方法没有任何操作,只需要覆盖intercetor()即可。
public abstract class AbstractInterceptor
implements Interceptor{ ... }
AbstractInterceptor
n 功能要求: 能够拦截Struts请求,并打印一些提示信息。
n 这里,我们继承 AbstractInterceptor,覆盖intercept()方法
public class LoginInteceptor extends AbstractInterceptor {
@Override //登录校验的拦截器
public String intercept(ActionInvocation ai) throws Exception {
System.out.println( 用户是否登录的拦截处理...... );
//如果用户操作的是要进行登录的操作,就继续进行
if(ai.getAction() instanceof UserLoginAction){return ai.invoke();
}else{//判断session中是否存在用户的信息,如果存在则用户已经登录
Map String,Object session = ai.getInvocationContext().getSession();//获取session
if(session.get( username )==null){
//如果用户没有登录,直接跳转到登录界面
return login ;}else{return ai.invoke();
}
}
}
}
当多个action中都使用到了相同视图,这时我们应该把result定义为全局视图。struts1中提供了全局forward,struts2中也提供了相似功能:
package ....
global-results
result name= message /message.jsp /result
/global-results
/package
Struts.xml文件的配置如下:
package name= ssh extends= struts-default namespace= /
!-- 声明自定义的拦截器 --
interceptors
interceptor name= UserLogin
class= com.redarmy.intercepter.LoginInteceptor /interceptor
/interceptors
!-- 声明全局结果 --
global-results
result name= login /user/login.jsp /result
/global-results
action name= listUser class= redarmy.shop.action.UserAction
method= listUsers
interceptor-ref name= UserLogin /interceptor-ref
!-- 当使用自己定义拦截器的时候,默认拦截器将不能使用,必须手动添加defaultStack的支持 --
interceptor-ref name= defaultStack /interceptor-ref
result name= success /user/listUser.jsp /result
result name= input /index.jsp /result
/action
/package
说明:
因为struts2中如文件上传,数据验证,封装请求参数到action等功能都是由系统默认的
defaultStack中的拦截器实现的,所以我们定义的拦截器需要引用系统默认的defaultStack,
这样应用才可以使用struts2框架提供的众多功能。
如果一个action中引入多个拦截器的时候,可以采用拦截器栈处理具体配置如下
package name= ssh extends= struts-default namespace= /
!-- 声明自定义的拦截器 --
interceptors
interceptor name= MyTimer“ class= com.redarmy.intercepter.MyTimerInterceptor
/interceptor
interceptor name= UserCheckLogin “ class=
com.redarmy.intercepter.LoginInteceptor /interceptor
!-- 定义拦截器的栈 --
interceptor-stack name= mystack
interceptor-ref name= MyTimer /interceptor-ref
interceptor-ref name= UserCheckLogin /interceptor-ref
!-- 添加Struts2默认拦截器的支持 --
interceptor-ref name= defaultStack /interceptor-ref
/interceptor-stack
/interceptors
action name= listUser class= redarmy.shop.action.UserAction “ method= listUsers
!-- 直接引用自己定义的拦截器栈 --
interceptor-ref name= mystack /interceptor-ref
result name= success /user/listUser.jsp /result
result name= input /index.jsp /result
/action
/package
如果希望包下的所有action都使用自定义的拦截器,可以通过 default-interceptor-ref name=“mystack”/ 把拦截器定义为默认拦截器。注意:每个包只能指定一个默认拦截器。另外,一旦我们为该包中的某个action显式指定了某个拦截器,则默认拦截器不会起作用。
在大部分应用里,随着应用规模的增加,系统中Action的数量也会大量增加,导致
struts.xml配置文件变得非常臃肿。为了避免struts.xml文件过于庞大、臃肿,提高
struts.xml文件的可读性,我们可以将一个struts.xml配置文件分解成多个配置文件,然
后在struts.xml文件中包含其他配置文件。下面的struts.xml通过 include 元素指定多个
配置文件:
?xml version= 1.0 encoding= UTF-8 ?
!DOCTYPE struts PUBLIC
-//Apache Software Foundation//DTD Struts Configuration 2.0//EN
struts
include file= redarmy\\shop\\action\\struts-zhuru.xml /include
include file= user.xml /include
/struts
margin-top: 0pt;
分享到:
Struts2中的依赖注入 |
Struts2的框架搭建及简单
2011-05-20 08:55
浏览 182
评论(0)
分类:企业架构
相关推荐
评论
发表评论
您还没有登录,请您登录后再发表评论