分類  >  Web前端 >

JavaWeb過濾器.監聽器.攔截器-原理&差異-個人總結(轉)

tags:    時間:2014-05-04 12:06:34
JavaWeb過濾器.監聽器.攔截器-原理&區別-個人總結(轉)

1、攔截器是基於java的反射機制的,而過濾器是基於函數回調 

2、過濾器依賴與servlet容器,而攔截器不依賴與servlet容器 

3、攔截器只能對action請求起作用,而過濾器則可以對幾乎所有的請求起作用 

4、攔截器可以訪問action上下文、值棧里的對象,而過濾器不能 

5、在action的生命周期中,攔截器可以多次被調用,而過濾器只能在容器初始化時被調用一次 

 

     攔截器 :是在面向切面編程的就是在你的service或者一個方法,前調用一個方法,或者在方法后調用一個方法比如動態代理就是攔截器的簡單實現,在你調用方法前列印出字元串(或者做其它業務邏輯的操作),也可以在你調用方法后列印出字元串,甚至在你拋出異常的時候做業務邏輯的操作。 

 

     1.Struts2攔截器是在訪問某個Action或Action的某個方法,欄位之前或之後實施攔截,並且Struts2攔截器是可插拔的,攔截器是AOP的一種實現。

     2. 攔截器棧(Interceptor Stack)。Struts2攔截器棧就是將攔截器按一定的順序聯結成一條鏈。在訪問被攔截的方法或欄位時,Struts2攔截器鏈中的攔截器就會按其之前定義的順序被調用。

 

 

 

     附:面向切面編程(AOP是Aspect Oriented Program的首字母縮寫) ,我們知道,面向對象的特點是繼承、多態和封裝。而封裝就要求將功能分散到不同的對象中去,這在軟體設計中往往稱為職責分配。實際上也就是說,讓不同的類設計不同的方法。這樣代碼就分散到一個個的類中去了。這樣做的好處是降低了代碼的複雜程度,使類可重用。

      但是人們也發現,在分散代碼的同時,也增加了代碼的重複性。什麼意思呢?比如說,我們在兩個類中,可能都需要在每個方法中做日誌。按面向對象的設計方法,我們就必須在兩個類的方法中都加入日誌的內容。也許他們是完全相同的,但就是因為面向對象的設計讓類與類之間無法聯繫,而不能將這些重複的代碼統一起來。

    也許有人會說,那好辦啊,我們可以將這段代碼寫在一個獨立的類獨立的方法里,然後再在這兩個類中調用。但是,這樣一來,這兩個類跟我們上面提到的獨立的類就有耦合了,它的改變會影響這兩個類。那麼,有沒有什麼辦法,能讓我們在需要的時候,隨意地加入代碼呢?這種在運行時,動態地將代碼切入到類的指定方法、指定位置上的編程思想就是面向切面的編程。 

      一般而言,我們管切入到指定類指定方法的代碼片段稱為切面,而切入到哪些類、哪些方法則叫切入點。有了AOP,我們就可以把幾個類共有的代碼,抽取到一個切片中,等到需要時再切入對象中去,從而改變其原有的行為。

這樣看來,AOP其實只是OOP的補充而已。OOP從橫向上區分出一個個的類來,而AOP則從縱向上向對象中加入特定的代碼。有了AOP,OOP變得立體了。如果加上時間維度,AOP使OOP由原來的二維變為三維了,由平面變成立體了。從技術上來說,AOP基本上是通過代理機制實現的。 

     AOP在編程歷史上可以說是里程碑式的,對OOP編程是一種十分有益的補充。

 

下面通過實例來看一下過濾器和攔截器的區別: 

 

使用攔截器進行/admin 目錄下jsp頁面的過濾 

 

<package name="newsDemo" extends="struts-default" 

        namespace="/admin"> 

        <interceptors> 

            <interceptor name="auth" class="com.test.news.util.AccessInterceptor" /> 

            <interceptor-stack name="authStack"> 

                <interceptor-ref name="auth" /> 

            </interceptor-stack> 

        </interceptors> 

        <!-- action --> 

        <action name="newsAdminView!*" class="newsAction" 

            method="{1}"> 

            <interceptor-ref name="defaultStack"/> 

            <interceptor-ref name="authStack"> 

            </interceptor-ref> 

 

下面是我實現的Interceptor class: 

 

package com.test.news.util; 

import java.util.Map; 

import com.opensymphony.xwork2.ActionContext; 

import com.opensymphony.xwork2.ActionInvocation; 

import com.opensymphony.xwork2.interceptor.AbstractInterceptor; 

import com.test.news.action.AdminLoginAction; 

/** 

* @author chaoyin */ 

 

public class AccessInterceptor extends AbstractInterceptor { 

 

    private static final long serialVersionUID = -4291195782860785705L; 

    @Override 

    public String intercept(ActionInvocation actionInvocation) throws Exception { 

         ActionContext actionContext = actionInvocation.getInvocationContext(); 

         Map session = actionContext.getSession();         

        //except login action 

         Object action = actionInvocation.getAction(); 

        if (action instanceof AdminLoginAction) { 

            return actionInvocation.invoke(); 

         } 

        //check session 

        if(session.get("user")==null ){ 

            return "logout"; 

         } 

        return actionInvocation.invoke();//go on 

     } 

       過濾器:是在java web中,你傳入的request,response提前過濾掉一些信息,或者提前設置一些參數,然後再傳入servlet或者struts的 action進行業務邏輯,比如過濾掉非法url(不是login.do的地址請求,如果用戶沒有登陸都過濾掉),或者在傳入servlet或者struts的action前統一設置字符集,或者去除掉一些非法字元。主要為了減輕伺服器負載,減少壓力。

 

使用過濾器進行/admin 目錄下jsp頁面的過濾,首先在web.xml進行過濾器配置: 

 

    <filter> 

        <filter-name>access filter</filter-name> 

        <filter-class> 

             com.test.news.util.AccessFilter 

        </filter-class> 

    </filter> 

    <filter-mapping> 

        <filter-name>access filter</filter-name> 

        <url-pattern>/admin/*</url-pattern> 

    </filter-mapping> 

 

下面是過濾的實現類: 

package com.test.news.util; 

import java.io.IOException; 

import javax.servlet.Filter; 

import javax.servlet.FilterChain; 

import javax.servlet.FilterConfig; 

import javax.servlet.ServletException; 

import javax.servlet.ServletRequest; 

import javax.servlet.ServletResponse; 

import javax.servlet.http.HttpServletRequest; 

import javax.servlet.http.HttpServletResponse; 

import javax.servlet.http.HttpSession; 

public class AccessFilter implements Filter { 

/** 

* @author chaoyin 

*/     

    public void destroy() { 

 

     } 

    public void doFilter(ServletRequest arg0, ServletResponse arg1, 

             FilterChain filterChain) throws IOException, ServletException { 

         HttpServletRequest request = (HttpServletRequest)arg0; 

         HttpServletResponse response = (HttpServletResponse)arg1; 

         HttpSession session = request.getSession(); 

        if(session.getAttribute("user")== null && request.getRequestURI().indexOf("login.jsp")==-1 ){ 

             response.sendRedirect("login.jsp"); 

            return ; 

         } 

         filterChain.doFilter(arg0, arg1); 

 

     } 

    public void init(FilterConfig arg0) throws ServletException { 

 

     } 

}

 

編程改變生活,改變人生

推薦閱讀文章

Bookmark the permalink ,來源:互聯網