實現session監聽器


監聽器概述   

1.Listener是Servlet的監聽器    
2.可以監聽客戶端的請求、服務端的操作等。   
3.通過監聽器,可以自動激發一些操作,如監聽在線用戶數量,當增加一個HttpSession時,給在線人數加1。   
4.編寫監聽器需要實現相應的接口   
5.編寫完成后在web.xml文件中配置一下,就可以起作用了   
6.可以在不修改現有系統基礎上,增加web應用程序生命周期事件的跟蹤   


常用的監聽接口   

Java代碼

  1. 1.ServletContextAttributeListener      
  2. 監聽對ServletContext屬性的操作,比如增加/刪除/修改      
  3. 2.ServletContextListener      
  4. 監聽ServletContext,當創建ServletContext時,激發 contextInitialized(ServletContextEvent sce)方法;當銷毀ServletContext時,激發contextDestroyed(ServletContextEvent sce)方法。      
  5. 3.HttpSessionListener      
  6. 監聽HttpSession的操作。當創建一個 Session時,激發session Created(SessionEvent se)方法;當銷毀一個Session時,激發sessionDestroyed (HttpSessionEvent se)方法。      
  7. 4.HttpSessionAttributeListener      
  8. 監聽HttpSession中的屬性的操作。當在Session增加一個屬性時,激發 attributeAdded(HttpSessionBindingEvent se) 方法;當在Session刪除一個屬性時,激發attributeRemoved(HttpSessionBindingEvent se)方法;當在Session屬性被重新設置時,激發attributeReplaced(HttpSessionBindingEvent se) 方法。    

1.ServletContextAttributeListener 監聽對ServletContext屬性的操作,比如增加/刪除/修改 2.ServletContextListener 監聽ServletContext,當創建ServletContext時,激發 contextInitialized(ServletContextEvent sce)方法;當銷毀ServletContext時,激發contextDestroyed(ServletContextEvent sce)方法。 3.HttpSessionListener 監聽HttpSession的操作。當創建一個 Session時,激發session Created(SessionEvent se)方法;當銷毀一個Session時,激發sessionDestroyed (HttpSessionEvent se)方法。 4.HttpSessionAttributeListener 監聽HttpSession中的屬性的操作。當在Session增加一個屬性時,激發 attributeAdded(HttpSessionBindingEvent se) 方法;當在Session刪除一個屬性時,激發attributeRemoved(HttpSessionBindingEvent se)方法;當在Session屬性被重新設置時,激發attributeReplaced(HttpSessionBindingEvent se) 方法。

使用范例:   
由監聽器管理共享數據庫連接   

生命周期事件的一個實際應用由context監聽器管理共享數據庫連接。在web.xml中如下定義監聽器:   
Java代碼

  1. <listener>      
  2.      <listener-class>XXX.MyConnectionManager</listener-class>      
  3. </listener>   

<listener> <listener-class>XXX.MyConnectionManager</listener-class> </listener>
server創建監聽器的實例,接受事件並自動判斷實現監聽器接口的類型。要記住的是由於監聽器是配置在部署描述符web.xml中,所以不需要改變任何代碼就可以添加新的監聽器。   
Java代碼

  1.      
  2. public class MyConnectionManager implements ServletContextListener{        
  3. public void contextInitialized(ServletContextEvent e) {       
  4.          Connection con = // create connection       
  5.          e.getServletContext().setAttribute("con", con);       
  6.      }        
  7.    public void contextDestroyed(ServletContextEvent e) {       
  8.          Connection con = (Connection) e.getServletContext().getAttribute("con");       
  9.         try {      
  10.            con.close();       
  11.          }       
  12.        catch (SQLException ignored) { } // close connection       
  13.      }       
  14. }   

public class MyConnectionManager implements ServletContextListener{ public void contextInitialized(ServletContextEvent e) { Connection con = // create connection e.getServletContext().setAttribute("con", con); } public void contextDestroyed(ServletContextEvent e) { Connection con = (Connection) e.getServletContext().getAttribute("con"); try { con.close(); } catch (SQLException ignored) { } // close connection } }    
監聽器保證每新生成一個servlet context都會有一個可用的數據庫連接,並且所有的連接對會在context關閉的時候隨之關閉。     

在web.xml中加入:   
Java代碼

  1. <listener><listener-class>servletlistener111111.SecondListener</listener-class> </listener>  

<listener><listener-class>servletlistener111111.SecondListener</listener-class> </listener>

================================================== 

關於用戶超時的例子: 

Java代碼

  1. public class OnlineUserListener implements HttpSessionListener {   
  2.     public void sessionCreated(HttpSessionEvent event) {   
  3.      }   
  4.     public void sessionDestroyed(HttpSessionEvent event) {   
  5.          HttpSession session = event.getSession();   
  6.          ServletContext application = session.getServletContext();   
  7.         // 取得登錄的用戶名   
  8.          String username = (String) session.getAttribute("username");   
  9.         // 從在線列表中刪除用戶名   
  10.          List onlineUserList = (List) application.getAttribute("onlineUserList");   
  11.          onlineUserList.remove(username);   
  12.          System.out.println(username + "超時退出。");   
  13.      }   
  14. }  

public class OnlineUserListener implements HttpSessionListener { public void sessionCreated(HttpSessionEvent event) { } public void sessionDestroyed(HttpSessionEvent event) { HttpSession session = event.getSession(); ServletContext application = session.getServletContext(); // 取得登錄的用戶名 String username = (String) session.getAttribute("username"); // 從在線列表中刪除用戶名 List onlineUserList = (List) application.getAttribute("onlineUserList"); onlineUserList.remove(username); System.out.println(username + "超時退出。"); }}

以下兩種情況下就會發生sessionDestoryed(會話銷毀)事件: 

1.執行session.invalidate()方法時。例如:request.getSession().invalidate(); 

2.如果用戶長時間沒有訪問服務器,超過了會話最大超時時間,服務器就會自動銷毀超時的session。會話超時時間可以在web.xml中進行設置。 

======================================== 

使用HttpSessionBindingListener 

HttpSessionBindingListener雖然叫做監聽器,但使用方法與HttpSessionListener完全不同。我們實際看一下它是如何使用的。

我們的OnlineUserBindingListener實現了HttpSessionBindingListener接口,接口中共定義了兩個方法:valueBound()和valueUnbound(),分別對應數據綁定,和取消綁定兩個事件。 

所謂對session進行數據綁定,就是調用session.setAttribute()把HttpSessionBindingListener保存進session中。我們在LoginServlet.java中進行這一步。 

// 把用戶名放入在線列表 
session.setAttribute("onlineUserBindingListener", new OnlineUserBindingListener(username)); 
       
這就是HttpSessionBindingListener和HttpSessionListener之間的最大區別:HttpSessionListener只需要設置到web.xml中就可以監聽整個應用中的所有session。 HttpSessionBindingListener必須實例化后放入某一個session中,才可以進行監聽。 

從監聽范圍上比較,HttpSessionListener設置一次就可以監聽所有session,HttpSessionBindingListener通常都是一對一的。 

正是這種區別成就了HttpSessionBindingListener的優勢,我們可以讓每個listener對應一個username,這樣就不需要每次再去session中讀取username,進一步可以將所有操作在線列表的代碼都移入listener,更容易維護。 

valueBound()方法的代碼如下: 

Java代碼

  1. public void valueBound(HttpSessionBindingEvent event) {   
  2.      HttpSession session = event.getSession();   
  3.      ServletContext application = session.getServletContext();   
  4.   
  5.     // 把用戶名放入在線列表   
  6.      List onlineUserList = (List) application.getAttribute("onlineUserList");   
  7.     // 第一次使用前,需要初始化   
  8.     if (onlineUserList == null) {   
  9.          onlineUserList = new ArrayList();   
  10.          application.setAttribute("onlineUserList", onlineUserList);   
  11.      }   
  12.      onlineUserList.add(this.username);   
  13. }  

public void valueBound(HttpSessionBindingEvent event) { HttpSession session = event.getSession(); ServletContext application = session.getServletContext(); // 把用戶名放入在線列表 List onlineUserList = (List) application.getAttribute("onlineUserList"); // 第一次使用前,需要初始化 if (onlineUserList == null) { onlineUserList = new ArrayList(); application.setAttribute("onlineUserList", onlineUserList); } onlineUserList.add(this.username);}
       
username已經通過構造方法傳遞給listener,在數據綁定時,可以直接把它放入用戶列表。 

與之對應的valueUnbound()方法,代碼如下: 
Java代碼

  1. public void valueUnbound(HttpSessionBindingEvent event) {   
  2.      HttpSession session = event.getSession();   
  3.      ServletContext application = session.getServletContext();   
  4.   
  5.     // 從在線列表中刪除用戶名   
  6.      List onlineUserList = (List) application.getAttribute("onlineUserList");   
  7.      onlineUserList.remove(this.username);   
  8.   
  9.      System.out.println(this.username + "退出。");   
  10. }  

public void valueUnbound(HttpSessionBindingEvent event) { HttpSession session = event.getSession(); ServletContext application = session.getServletContext(); // 從在線列表中刪除用戶名 List onlineUserList = (List) application.getAttribute("onlineUserList"); onlineUserList.remove(this.username); System.out.println(this.username + "退出。");}
       
這里可以直接使用listener的username操作在線列表,不必再去擔心session中是否存在username。 

valueUnbound的觸發條件是以下三種情況: 
Java代碼

  1. 1.執行session.invalidate()時。   
  2.   
  3. 2.session超時,自動銷毀時。   
  4.   
  5. 3.執行session.setAttribute("onlineUserListener", "其他對象");或session.removeAttribute("onlineUserListener");將listener從session中刪除時。  

1.執行session.invalidate()時。2.session超時,自動銷毀時。3.執行session.setAttribute("onlineUserListener", "其他對象");或session.removeAttribute("onlineUserListener");將listener從session中刪除時。

因此,只要不將listener從session中刪除,就可以監聽到session的銷毀。


免責聲明!

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



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