日韩无码专区无码一级三级片|91人人爱网站中日韩无码电影|厨房大战丰满熟妇|AV高清无码在线免费观看|另类AV日韩少妇熟女|中文日本大黄一级黄色片|色情在线视频免费|亚洲成人特黄a片|黄片wwwav色图欧美|欧亚乱色一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
Struts2教程9:實現(xiàn)自已的攔截器

【相關(guān)文章】

  1. Struts2教程1:***個Struts2程序
  2. Struts2教程2:處理一個form多個submit
  3. Struts2教程3:struts.xml常用配置解析
  4. Struts2教程4:使用validate方法驗證數(shù)據(jù)
  5. Struts2教程5:使用Validation框架驗證數(shù)據(jù)
  6. Struts2教程6:在Action類中獲得HttpServletResponse對象
  7. Struts2教程7:上傳任意多個文件
  8. Struts2教程8:攔截器概述
  9. Struts2教程10:國際化


在上一篇中介紹了Struts2攔截器的原理,在這一篇中我們將學(xué)習(xí)一下如何編寫自己的攔截器。

攔截器的實現(xiàn)

實現(xiàn)一個攔截器非常簡單。實際上,一個攔截器就是一個普通的類,只是這個類必須實現(xiàn)com.opensymphony.xwork2.interceptor.Interceptor接口。Interceptor接口有如下三個方法:

publicinterfaceInterceptorextendsSerializable
{
  voiddestroy();
  voidinit();
  Stringintercept(ActionInvocationinvocation)throwsException;
}

其中init和destroy方法只在攔截器加載和釋放(都由Struts2自身處理)時執(zhí)行一次。而intercept方法在每次訪問動作時都會被調(diào)用。Struts2在調(diào)用攔截器時,每個攔截器類只有一個對象實例,而所有引用這個攔截器的動作都共享這一個攔截器類的對象實例,因此,在實現(xiàn)Interceptor接口的類中如果使用類變量,要注意同步問題。

下面我們來實現(xiàn)一個簡單的攔截器,這個攔截器通過請求參數(shù)action指定一個攔截器類中的方法,并調(diào)用這個方法(我們可以使用這個攔截器對某一特定的動作進行預(yù)處理)。如果方法不存在,或是action參數(shù)不存在,則繼續(xù)執(zhí)行下面的代碼。如下面的URL:

http://localhost:8080/struts2/test/interceptor.action?action=test

訪問上面的url后,攔截器會就會調(diào)用攔截器中的test方法,如果這個方法不存在,則調(diào)用invocation.invoke方法,invoke方法和Servlet過濾器中調(diào)用FilterChain.doFilter方法類似,如果在當(dāng)前攔截器后面還有其他的攔截器,則invoke方法就是調(diào)用后面攔截器的intercept方法,否則,invoke會調(diào)用Action類的execute方法(或其他的執(zhí)行方法)。

下面我們先來實現(xiàn)一個攔截器的父類ActionInterceptor。這個類主要實現(xiàn)了根據(jù)action參數(shù)值來調(diào)用方法的功能,代碼如下:

packageinterceptor;

importcom.opensymphony.xwork2.ActionInvocation;
importcom.opensymphony.xwork2.interceptor.Interceptor;
importjavax.servlet.http.*;
importorg.apache.struts2.*;
publicclassActionInterceptorimplementsInterceptor
{
  protectedfinalStringINVOKE="##invoke";
 
  publicvoiddestroy()
  {
    System.out.println("destroy");
  }
  publicvoidinit()
  {
    System.out.println("init");
  }
  publicStringintercept(ActionInvocationinvocation)throwsException
  {  
    HttpServletRequestrequest=ServletActionContext.getRequest();
    Stringaction=request.getParameter("action");
    System.out.println(this.hashCode());
    if(action!=null)
    {
      try
      {
        java.lang.reflect.Methodmethod=this.getClass().getMethod(action);
        Stringresult=(String)method.invoke(this);
        if(result!=null)
        {
          if(!result.equals(INVOKE))
            returnresult;
        }
        else
          returnnull;
      }
      catch(Exceptione)
      {
      }
    }
    returninvocation.invoke();
  }
}

從上面代碼中的intercept方法可以看出,在調(diào)用action所指定的方法后,來判斷返回值。可能發(fā)生的情況有三種:

1. 返回值為null,執(zhí)行return null。

2. 返回值為INVOKE,執(zhí)行return invockation.invoke()。

3. 其他情況,執(zhí)行return result。result表示指定方法的返回值,如上面代碼所示。

在實現(xiàn)完上面的攔截器父類后,任何繼承于ActionInterceptor類的攔截器都可以自動根據(jù)action的參數(shù)值調(diào)用自身的相應(yīng)方法。下面我們來實現(xiàn)一個擁有兩個動作方法test和print的攔截器類。代碼如下:

packageinterceptor;

importjavax.servlet.http.HttpServletResponse;
importorg.apache.struts2.ServletActionContext;
publicclassMultiMethodInterceptorextendsActionInterceptor
{
  publicStringtest()throwsException
  {
    HttpServletResponseresponse=ServletActionContext.getResponse();
    response.getWriter().println("invoketest");
    returnthis.INVOKE;
  }
  publicStringprint()throwsException
  {
    HttpServletResponseresponse=ServletActionContext.getResponse();
    response.getWriter().println("invokeprint");
    returnnull;
  }
}

est方法返回了INVOKE,因此,在執(zhí)行完這個方法后,Struts2會接著調(diào)用其他攔截器的intercept方法或Action類的execute方法。而print方法在執(zhí)行完后,只是返回了null,而不再調(diào)用其他的方法了,也就是訪問如下的url時,動作的execute方法將不會執(zhí)行:

http://localhost:8080/struts2/test/ddd.action?action=print

下面我們來實現(xiàn)一個Action類,代碼如下:

packageaction;

importorg.apache.struts2.*;
importcom.opensymphony.xwork2.ActionSupport;
publicclassInterceptorActionextendsActionSupport
{
  publicStringabcd()throwsException
  {
    ServletActionContext.getResponse().getWriter()
        .println("invokeabcd");
    returnnull;
  }
}

#p#

在這個Action類中,只有一個abcd方法,實際上,這個方法相當(dāng)于execute方法,在下面會設(shè)置動作的method屬性為abcd。下面我們來在struts.xml中定義攔截器類和動作,代碼如下:

<?xmlversion="1.0"encoding="UTF-8"?>
<!DOCTYPEstrutsPUBLIC
  "-//ApacheSoftwareFoundation//DTDStrutsConfiguration2.0//EN"
  "http://struts.apache.org/dtds/struts-2.0.dtd">
<struts>
 ?。紁ackagename="demo"extends="struts-default"namespace="/test">
   ?。糹nterceptors>
      <interceptorname="method"class="interceptor.MultiMethodInterceptor"/>
       ?。糹nterceptor-stackname="methodStack">
          <interceptor-refname="method"/>
         ?。糹nterceptor-refname="defaultStack"/>
        </interceptor-stack>
   ?。?interceptors>
   ?。糰ctionname="interceptor"class="action.InterceptorAction"method="abcd">
     ?。糹nterceptor-refname="methodStack"/>
   ?。?action>
 ?。?package>
</struts>

在配置上面的methodStack攔截器時要注意,***在后面引用defaultStack,否則很多通過攔截器提供的功能將失去。

現(xiàn)在訪問如下的URL:

http://localhost:8080/struts2/test/ddd.action?action=test

在瀏覽器中將會出現(xiàn)如下的字符串:

invoke test

invoke abcd

而如果訪問http://localhost:8080/struts2/test/ddd.action?action=print,將會只出現(xiàn)如下的字符串:

invoke print

大家可以看出,訪問這個url時并沒有調(diào)用abcd方法。如果隨便指定的action值的話,則只調(diào)用abcd方法,如訪問http://localhost:8080/struts2/test/ddd.action?action=aaa,就只會輸出invoke abcd。

攔截器的參數(shù)

我們在使用很多Struts2內(nèi)置的攔截器時會發(fā)現(xiàn)有很多攔截器都帶參數(shù),當(dāng)然。我們自己做的攔截器也可以加上同樣的參數(shù)。有兩個參數(shù)比較常用,這兩個參數(shù)是includeMethods和excludeMethods,其中includeMethods指定了攔截器要調(diào)用的Action類的執(zhí)行方法(默認(rèn)是execute),也就是說,只有在includeMethods中指定的方法才會被Struts2調(diào)用,而excludeMethods恰恰相反,在這個參數(shù)中指定的執(zhí)行方法不會被Struts2調(diào)用。如果有多個方法,中間用逗號(,)分隔。在Struts2中提供了一個抽象類來處理這兩個參數(shù)。這個類如下:

com.opensymphony.xwork2.interceptor.MethodFilterInterceptor

如有繼承于這個類的攔截器類都會自動處理includeMethods和excludeMethods參數(shù),如下面的攔截器類所示:

packageinterceptor;

importcom.opensymphony.xwork2.ActionInvocation;
importcom.opensymphony.xwork2.interceptor.*;
publicclassMyFilterInterceptorextendsMethodFilterInterceptor
{
  privateStringname;
  publicStringgetName()
  {
    returnname;
  }
  publicvoidsetName(Stringname)
  {
    this.name=name;
  }
  @Override
  protectedStringdoIntercept(ActionInvocationinvocation)throwsException
  {
    System.out.println("doIntercept");
    System.out.println(name);
    returninvocation.invoke();
  }
}

MethodFilterInterceptor的子類需要實現(xiàn)doIntercept方法(相當(dāng)于Interceptor的intercept方法),如上面代碼所示。在上面的代碼中還有一個name屬性,是為了讀取攔截器的name屬性而設(shè)置的,如下面的配置代碼所示:

<?xmlversion="1.0"encoding="UTF-8"?>

<!DOCTYPEstrutsPUBLIC

  "-//ApacheSoftwareFoundation//DTDStrutsConfiguration2.0//EN"

  "http://struts.apache.org/dtds/struts-2.0.dtd">

<struts>

 ?。紁ackagename="demo"extends="struts-default"namespace="/test">

    <interceptors>

     ?。糹nterceptorname="method"class="interceptor.MultiMethodInterceptor"/>

        <interceptorname="filter"

          class="interceptor.MyFilterInterceptor">

         ?。紁aramname="includeMethods">abcd</param>

         ?。紁aramname="name">中國</param>

        </interceptor>

       ?。糹nterceptor-stackname="methodStack">

         ?。糹nterceptor-refname="method"/>

         ?。糹nterceptor-refname="filter"/>

          <interceptor-refname="defaultStack"/>

       ?。?interceptor-stack>

   ?。?interceptors>

    <actionname="interceptor"class="action.InterceptorAction"method="abcd">

     ?。糹nterceptor-refname="methodStack"/>

    </action>

 ?。?package>

</struts>

再次訪問http://localhost:8080/struts2/test/ddd.action?action=test, Struts2就會調(diào)用MyFilterInterceptor的doIntercept方法來輸出name屬性值。如果將上面的includeMethods參數(shù)值中的abcd去掉,則Action類的abcd方法不會被執(zhí)行。


本文題目:Struts2教程9:實現(xiàn)自已的攔截器
轉(zhuǎn)載注明:http://m.5511xx.com/article/copehse.html