目录

一、Struts2拦截器Interceptor:

   (一)struts2拦截器的使用

   (二)struts内建拦截器

   (三)拦截器使用

二、拦截器的具体实现原理(转)

一、Struts2拦截器Interceptor

(一)struts2拦截器的使用

1、拦截器是struts2扩展Action的机制,拦截器就是一个类,实现了Interceptor 接口的一个类。通过拦截器做一些系统级通用的业务逻辑。如:记录日志、控制权限、控制事务

2Interceptor的使用步骤

  a)创建拦截器组件,实现Interceptor接口

  b)在struts.xml注册这个拦截器

  c)在struts.xml中,给action指定拦截器

  d)若Action中引用了自定义的Interceptor,那么默认的Interceptor就不会执行。

3、使用方式

<struts>
  <package name="demo" namespace="/demo"      extends="struts-default">
  <!--定义拦截器组 -->
   <interceptorsname="myInterceptors">
       <!--定义自己的拦截器 -->
       <interceptor name="myInterceptor"class="com.tarena.interceptor.MyInterceptor"></interceptor>
       </interceptors>
       <!--定义拦截器栈,可以作为一个普通的拦截器,
被Action引用 -->
       <interceptor-stackname="myInterceptorStack">
            <!--引用拦截器 -->
            <interceptor-refname="myInterceptor"></interceptor-ref>
         </interceptor-stack>
         <actionname="myAction"
                   class="com.tarena.action.MyAction">
                   <!--引用拦截器 -->
                   <intercepter-refname="myInterceptorStack"></interceptor-ref>
         </action>
  </package>
</struts>

4Struts中拦截器和Servlet中的Filter有类似的功能,Struts中的拦截器的实现也是一样,在具体Action的被调用之前或之后可以做一些操作,采用配置化的方法进行管理,使用起来比较简单。

144133709.jpg

(二)struts内建拦截器

alias:实现在不同请求中相似参数别名的转换。

autowiring:这是个自动装配的拦截器,主要用于当Struts2Spring整合时,Struts2可以使用自动装配的方式来访问Spring容器中的Bean

chain:构建一个Action链,使当前Action可以访问前一个Action的属性,一般和<resulttype="chain" .../>一起使用。

conversionError:这是一个负责处理类型转换错误的拦截器,它负责将类型转换错误从ActionContext中取出,并转换成ActionFieldError错误。

createSession:该拦截器负责创建一个HttpSession对象,主要用于那些需要有HttpSession对象才能正常工作的拦截器中。

debugging:当使用Struts2的开发模式时,这个拦截器会提供更多的调试信息。

execAndWait:后台执行Action,负责将等待画面发送给用户。

exception:这个拦截器负责处理异常,它将异常映射为结果。

fileUpload:这个拦截器主要用于文件上传,它负责解析表单中文件域的内容。

i18n:这是支持国际化的拦截器,它负责把所选的语言、区域放入用户Session中。

logger:这是一个负责日志记录的拦截器,主要是输出Action的名字。

model-driven:这是一个用于模型驱动的拦截器,当某个Action类实现了ModelDriven接口时,它负责把getModel()方法的结果堆入ValueStack中。

scoped-model-driven:如果一个Action实现了一个ScopedModelDriven接口,该拦截器负责从指定生存范围中找出指定的Modol,并将通过setModel方法将该Model传给Action实例。

params:这是最基本的一个拦截器,它负责解析HTTP请求中的参数,并将参数值设置成Action对应的属性值。

prepare:如果action实现了Preparable接口,将会调用该拦截器的prepare()方法。

static-params:这个拦截器负责将xml<action>标签下<param>标签中的参数传入action

scope:这是范围转换拦截器,它可以将Action状态信息保存到HttpSession范围,或者保存到ServletContext范围内。

servlet-config:如果某个Action需要直接访问Servlet API,就是通过这个拦截器实现的。

注意:尽量避免在Action中直接访问Servlet API,这样会导致ActionServlet的高耦合。

roles:这是一个JAASJava Authentication andAuthorization ServiceJava授权和认证服务)拦截器,只有当浏览者取得合适的授权后,才可以调用被该拦截器拦截的Action

timer:这个拦截器负责输出Action的执行时间,这个拦截器在分析该Action的性能瓶颈时比较有用。

token:这个拦截器主要用于阻止重复提交,它检查传到Action中的token,从而防止多次提交。

token-session:这个拦截器的作用与前一个基本类似,只是它把token保存在HttpSession中。

validation:通过执行在xxxAction-validation.xml中定义的校验器,从而完成数据校验。

workflow:这个拦截器负责调用Action类中的validate方法,如果校验失败,则返回input的逻辑视图。

大部分时候,开发者无需手动控制这些拦截器,因为struts-default.xml文件中已经配置了这些拦截器,只要我们定义的包继承了系统的struts-default包,就可以直接使用这些拦截器.

(三)拦截器使用

继承AbstractInterceptor,重写interceptorActionInvocationinvocation)方法

继承MethodFilterInterceptor,重写doInterceptorActionInvocationinvocation)方法配置参数时可指定过滤的方法,或排除的方法。

public class FirstInterceptor implements Interceptor {
    public void destroy() {
    }
    public void init() {
    }
    public String intercept(ActionInvocation ai) throws Exception {
        // 调用Action之前,执行的代码
        System.out.println("调用前:FirstInterceptor.intercept()...");
        // 调用对应的Action
        String result = ai.invoke();
        // 调用Action之后,执行的代码
        System.out.println("调用后:FirstInterceptor.intercept()...");
        return "error";
    }
}

struts.xml

<package name="demo" namespace="/demo" extends="struts-default">
        <!-- 一个Struts2拦截器的小例子 -->
        <interceptors>
            <!-- 自定义拦截器 -->
            <interceptor name="firstInterceptor" class="com.tarena.interceptor.FirstInterceptor"></interceptor>
            <interceptor name="secondInterceptor" class="com.tarena.interceptor.SecondInterceptor"></interceptor>
                                     
            <!-- 自定义拦截器栈,拦截器打包 -->
            <interceptor-stack name="iner">
                <interceptor-ref name="firstInterceptor"></interceptor-ref>
                <interceptor-ref name="secondInterceptor"></interceptor-ref>
            </interceptor-stack>
        </interceptors>     
                                 
        <action name="some" class="com.tarena.action.SomeAction">
            <!-- <interceptor-ref name="firstInterceptor"></interceptor-ref> -->
            <!-- <interceptor-ref name="secondInterceptor"></interceptor-ref> -->
            <!-- 一个个的使用拦截器麻烦,可以使用拦截器栈(一组拦截器) -->
            <interceptor-ref name="iner"></interceptor-ref>
            <result name="success">/WEB-INF/ok.jsp</result>
            <result name="error">/WEB-INF/error.jsp</result>
        </action>
    </package>

Action:

public String execute() {
        System.out.println("SomeAction.execute()...");
        return "success";
    }