java實現在線人數統計 (2011-05-30 15:10)

原來的OA項目有這樣的一個需求,開始查找了些資料用了HttpSessionListener接口來實現,具體的方法是:首先通過實現HttpSessionListener接口,具體代碼如下:
import java.util.List;
04 import javax.servlet.ServletContext;
05 import javax.servlet.http.HttpSession;
06 import javax.servlet.http.HttpSessionEvent;
07 import javax.servlet.http.HttpSessionListener;
08 /**
09  * @author 版本
10  */
11 public class OnlineUserListener implements HttpSessionListener {
12   
13     public void sessionCreated(HttpSessionEvent event) {
14         System.out.println("新建session:"+event.getSession().getId());
15     }
16     public void sessionDestroyed(HttpSessionEvent event) {
17         HttpSession session = event.getSession();
18         ServletContext application = session.getServletContext();
19         // 取得登錄的用戶名
20         String username = (String) session.getAttribute("username");
21         // 從在線列表中刪除用戶名
22         List onlineUserList = (List) application.getAttribute("onlineUserList");
23         onlineUserList.remove(username);
24         System.out.println(username+"已經退出!");
25     }
26 }
 
然後再配置監聽,

web.xml配置:

1 <listener>
2   <listener-class>com.test.OnlineUserListener</listener-class>
3 </listener>

一旦監聽器發現調用了sessionDestoryed方法就會把其用戶從在線人數中delete,在下面兩種情況下會發生sessionDestoryed事件

a.執行session.invalidate()方法時

logout.jsp中調用了 session.invalidate()方法

b.session會話超時

session的默認超時事件是30分鐘,30分鐘後自動銷燬session,這個時間也可以根據實際需要進行設置。

後來發現這種方式經常會出現數據丟失的情況,後來又發現通過實現HttpSessionBindingListener接口來實現,通過監聽session的新建和銷燬來控制也可以,具體方法如下:

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

新建類OnlineUserBindingListener,實現HttpSessionBindingListener接口,構造方法傳入username參數,HttpSessionBindingListener內有兩個方法valueBound(HttpSessionBindingEvent event)和valueUnbound(HttpSessionBindingEvent event),前者爲數據綁定,後者爲取消綁定

所謂對session進行數據綁定,就是調用session.setAttribute()把HttpSessionBindingListener保存進session中。

在login.jsp中做這一步:

01 <%@page import="com.test.OnlineUserBindingListener"%>
02 <%@ page contentType="text/html;charset=utf-8"%>
03 <%@ page import="java.util.*"%>
04 <%
05     request.setCharacterEncoding("UTF-8");
06     // 取得登錄的用戶名
07     String username = request.getParameter("username");
08        // 把用戶名放入在線列表
09     session.setAttribute("onlineUserBindingListener", new OnlineUserBindingListener(username));
10     // 成功
11     response.sendRedirect("result.jsp");
12 %>

這就是HttpSessionBindingListener和HttpSessionListener之間的最大區別:HttpSessionListener只需要設置到web.xml中就可以監聽整個應用中的所有session。HttpSessionBindingListener必須實例化後放入某一個session中,纔可以進行監聽。

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

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

HttpSessionBindingListener代碼如下:

01 package com.test;
02   
03 import java.util.ArrayList;
04 import java.util.List;
05 import javax.servlet.ServletContext;
06 import javax.servlet.http.HttpSession;
07 import javax.servlet.http.HttpSessionBindingEvent;
08 import javax.servlet.http.HttpSessionBindingListener;
09   
10 public class OnlineUserBindingListener implements HttpSessionBindingListener {
11     String username;
12       
13     public OnlineUserBindingListener(String username){
14         this.username=username;
15     }
16     public void valueBound(HttpSessionBindingEvent event) {
17         HttpSession session = event.getSession();
18         ServletContext application = session.getServletContext();
19         // 把用戶名放入在線列表
20         List onlineUserList = (List) application.getAttribute("onlineUserList");
21         // 第一次使用前,需要初始化
22         if (onlineUserList == null) {
23             onlineUserList = new ArrayList();
24             application.setAttribute("onlineUserList", onlineUserList);
25         }
26         onlineUserList.add(this.username);
27     }
28   
29     public void valueUnbound(HttpSessionBindingEvent event) {
30         HttpSession session = event.getSession();
31         ServletContext application = session.getServletContext();
32   
33         // 從在線列表中刪除用戶名
34         List onlineUserList = (List) application.getAttribute("onlineUserList");
35         onlineUserList.remove(this.username);
36         System.out.println(this.username + "退出。");
37   
38     }
39   
40 }

這裏可以直接使用listener的username操作在線列表,不必再去擔心session中是否存在username。

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

a.執行session.invalidate()時。

b.session超時,自動銷燬時。

c.執行session.setAttribute("onlineUserListener", "其他對象");或session.removeAttribute("onlineUserListener");將listener從session中刪除時。

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

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章