Java 過濾器、監聽器、攔截器的區別


原文:http://www.360doc.com/content/10/0601/09/495229_30616324.shtml

1.過濾器

Servlet中的過濾器Filter是實現了javax.servlet.Filter接口的服務器端程序,主要的用途是過濾字符編碼、做一些業務邏輯判斷等。其工作原理是,只要你在web.xml文件配置好要攔截的客戶端請求,它都會幫你攔截到請求,此時你就可以對請求或響應(Request、Response)統一設置編碼,簡化操作;同時還可進行邏輯判斷,如用戶是否已經登陸、有沒有權限訪問該頁面等等工作。它是隨你的web應用啟動而啟動的,只初始化一次,以后就可以攔截相關請求,只有當你的web應用停止或重新部署的時候才銷毀,以下通過過濾編碼的代碼示例來了解它的使用:

MyCharsetFilter.java 編碼過濾器
package ...;
import ...;
// 主要目的:過濾字符編碼;其次,做一些應用邏輯判斷等.
// Filter跟web應用一起啟動
// 當web應用重新啟動或銷毀時,Filter也被銷毀
public class MyCharsetFilter implements Filter {
     private FilterConfig config = null;
     public void destroy() {
         System.out.println("MyCharsetFilter准備銷毀...");
     }
 
     public void doFilter(ServletRequest arg0, ServletResponse arg1, FilterChain chain) throws IOException, ServletException {
         // 強制類型轉換
         HttpServletRequest request = (HttpServletRequest)arg0;
         HttpServletResponse response = (HttpServletResponse)arg1;
         // 獲取web.xm設置的編碼集,設置到Request、Response中         request.setCharacterEncoding(config.getInitParameter("charset"));          response.setContentType(config.getInitParameter("contentType"));          response.setCharacterEncoding(config.getInitParameter("charset"));         
        // 將請求轉發到目的地
         chain.doFilter(request, response);
     }
 
     public void init(FilterConfig arg0) throws ServletException {
         this.config = arg0;
         System.out.println("MyCharsetFilter初始化...");
     }
 }

以下是 MyCharsetFilter.java 在web.xml 中配置: 

 <filter>
       <filter-name>filter</filter-name>
       <filter-class>dc.gz.filters.MyCharsetFilter</filter-class>
       <init-param>
           <param-name>charset</param-name>
           <param-value>UTF-8</param-value>
       </init-param>
       <init-param>
           <param-name>contentType</param-name>
           <param-value>text/html;charset=UTF-8</param-value>
       </init-param>
   </filter>
   <filter-mapping>
       <filter-name>filter</filter-name>
       <!-- * 代表截獲所有的請求  或指定請求/test.do  /xxx.do -->
       <url-pattern>/*</url-pattern>
   </filter-mapping>

     以上的例子簡單的說明了Filter的使用,具體其他的應用可以看具體的場景。 

2.監聽器

現在來說說Servlet的監聽器Listener,它是實現了javax.servlet.ServletContextListener 接口的服務器端程序,它也是隨web應用的啟動而啟動,只初始化一次,隨web應用的停止而銷毀。主要作用是: 做一些初始化的內容添加工作、設置一些基本的內容、比如一些參數或者是一些固定的對象等等。下面利用監聽器對數據庫連接池DataSource的初始化演示它的使用: 

MyServletContextListener.java
 package dc.gz.listeners;
 import javax.servlet.ServletContext;
 import javax.servlet.ServletContextEvent;
 import javax.servlet.ServletContextListener;
 import org.apache.commons.dbcp.BasicDataSource;
 
  /**
  * Web應用監聽器
  */
 public class MyServletContextListener implements ServletContextListener {  
     // 應用監聽器的銷毀方法
     public void contextDestroyed(ServletContextEvent event) {
         ServletContext sc = event.getServletContext();
         // 在整個web應用銷毀之前調用,將所有應用空間所設置的內容清空
         sc.removeAttribute("dataSource");
        System.out.println("銷毀工作完成...");
     }
 
     // 應用監聽器的初始化方法
     public void contextInitialized(ServletContextEvent event) {
         // 通過這個事件可以獲取整個應用的空間
         // 在整個web應用下面啟動的時候做一些初始化的內容添加工作
         ServletContext sc = event.getServletContext();
         // 設置一些基本的內容;比如一些參數或者是一些固定的對象
         // 創建DataSource對象,連接池技術 dbcp
         BasicDataSource bds = new BasicDataSource();
         bds.setDriverClassName("com.mysql.jdbc.Driver");                       bds.setUrl("jdbc:mysql://localhost:3306/hibernate");
         bds.setUsername("root");
         bds.setPassword("root");
         bds.setMaxActive(10);//最大連接數
         bds.setMaxIdle(5);//最大管理數
         //bds.setMaxWait(maxWait); 最大等待時間
         // 把 DataSource 放入ServletContext空間中,
         // 供整個web應用的使用(獲取數據庫連接)
         sc.setAttribute("dataSource", bds);
         System.out.println("應用監聽器初始化工作完成...");
         System.out.println("已經創建DataSource...");
     }
 } 

web.xml中配置如下,很簡單:

 <!-- 配置應用監聽器  -->
   <listener>
       <listener-class>dc.gz.listeners.MyServletContextListener</listener-class>
   </listener>
 
    這樣配置好了之后,以后在web應用中就可以通過ServletContext取得BasicDataSource對象,從而獲取與數據庫的連接,提高性能,方便使用。

3.攔截器

攔截器是在面向切面編程中應用的,就是在你的service或者一個方法前調用一個方法,或者在方法后調用一個方法。是基於JAVA的反射機制。攔截器不是在web.xml,比如struts在struts.xml中配置,

public Object invoke(Object proxy, Method method, Object[] args) throws Throwable
{
Object result = null;
System.out.println("before invoke method :" + method.getName());
result = method.invoke(this.targetObj, args);
System.out.println("after invoke method : " + method.getName());
return result;
}

總結:

1.過濾器:所謂過濾器顧名思義是用來過濾的,在java web中,你傳入的request,response提前過濾掉一些信息,或者提前設置一些參數,然后再傳入servlet或者struts的action進行業務邏輯,比如過濾掉非法url(不是login.do的地址請求,如果用戶沒有登陸都過濾掉),或者在傳入servlet或者struts的action前統一設置字符集,或者去除掉一些非法字符(聊天室經常用到的,一些罵人的話)。filter 流程是線性的, url傳來之后,檢查之后,可保持原來的流程繼續向下執行,被下一個filter, servlet接收等.

2.監聽器:這個東西在c/s模式里面經常用到,他會對特定的事件產生產生一個處理。監聽在很多模式下用到。比如說觀察者模式,就是一個監聽來的。又比如struts可以用監聽來啟動。Servlet監聽器用於監聽一些重要事件的發生,監聽器對象可以在事情發生前、發生后可以做一些必要的處理。

3.java的攔截器 主要是用在插件上,擴展件上比如 hivernate spring struts2等 有點類似面向切片的技術,在用之前先要在配置文件即xml文件里聲明一段的那個東西。

面向切面編程(AOP是Aspect Oriented Program的首字母縮寫) ,我們知道,面向對象的特點是繼承、多態和封裝。而封裝就要求將功能分散到不同的對象中去,這在軟件設計中往往稱為職責分配。實際上也就是說,讓不同的類設計不同的方法。這樣代碼就分散到一個個的類中去了。這樣做的好處是降低了代碼的復雜程度,使類可重用。
      但是人們也發現,在分散代碼的同時,也增加了代碼的重復性。什么意思呢?比如說,我們在兩個類中,可能都需要在每個方法中做日志。按面向對象的設計方法,我們就必須在兩個類的方法中都加入日志的內容。也許他們是完全相同的,但就是因為面向對象的設計讓類與類之間無法聯系,而不能將這些重復的代碼統一起來。

    也許有人會說,那好辦啊,我們可以將這段代碼寫在一個獨立的類獨立的方法里,然后再在這兩個類中調用。但是,這樣一來,這兩個類跟我們上面提到的獨立的類就有耦合了,它的改變會影響這兩個類。那么,有沒有什么辦法,能讓我們在需要的時候,隨意地加入代碼呢?這種在運行時,動態地將代碼切入到類的指定方法、指定位置上的編程思想就是面向切面的編程。 
      一般而言,我們管切入到指定類指定方法的代碼片段稱為切面,而切入到哪些類、哪些方法則叫切入點。有了AOP,我們就可以把幾個類共有的代碼,抽取到一個切片中,等到需要時再切入對象中去,從而改變其原有的行為。
這樣看來,AOP其實只是OOP的補充而已。OOP從橫向上區分出一個個的類來,而AOP則從縱向上向對象中加入特定的代碼。有了AOP,OOP變得立體了。如果加上時間維度,AOP使OOP由原來的二維變為三維了,由平面變成立體了。從技術上來說,AOP基本上是通過代理機制實現的。 
     AOP在編程歷史上可以說是里程碑式的,對OOP編程是一種十分有益的補充。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM