用HttpSessionListener與HttpSessionBindingListener實現在線人數統計[轉]

 在線人數統計方面的實現,上網找了下這方面的知識,最初我的想法是, 管理session,如果session銷燬了就減少,如果登陸用戶了就新增一個,但是如果是用戶非法退出,如:未註銷,關閉瀏覽器等,這個用戶的 session是管理不到的,最後決定用HttpSessionListener接口或HttpSessionBindingListener接口來實 現,通過監聽session的新建和銷燬來控制,詳細如下。

先添加登陸的頁面index.jsp

01 <%@ page contentType="text/html;charset=utf-8"%>
02 < html >
03 < head >
04 < title >test</ title >             
05 </ head >
06 < body >
07 < form action = "login.jsp" method = "post" >
08      用戶名:< input type = "text" name = "username" />
09      < br />
10      < input type = "submit" value = "登錄" />
11 </ form >
12 </ body >
13 </ html >

點擊登陸後跳轉的login.jsp(爲了方便,用jsp做servlet,同學們用的時候記得改過來)

01 <%@ page contentType="text/html;charset=utf-8"%>
02 <%@ page import="java.util.*"%>
03 <%
04      request.setCharacterEncoding("UTF-8");
05      // 取得登錄的用戶名
06      String username = request.getParameter("username");
07      // 把用戶名保存進session
08      session.setAttribute("username", username);
09      // 把用戶名放入在線列表
10      List onlineUserList = (List) application.getAttribute("onlineUserList");
11      // 第一次使用前,需要初始化
12      if (onlineUserList == null) {
13          onlineUserList = new ArrayList();
14          application.setAttribute("onlineUserList", onlineUserList);
15      }
16      onlineUserList.add(username);
17      // 成功
18      response.sendRedirect("result.jsp");
19 %>

登陸成功跳轉到顯示頁面result.jsp

1 <%@ page contentType="text/html;charset=utf-8"%>
2 <%@ page isELIgnored="false"%>
3 <%@page import="java.util.List"%>
01 < h3 >您好:${username} [< a href = "logout.jsp" >註銷</ a >]</ h3 >
02 當前在線用戶:
03 < table >
04 <%
05      List onlineUserList = (List) application.getAttribute("onlineUserList");
06      for (int i = 0; i < onlineUserList.size (); i++) {
07      String onlineUsername = (String) onlineUserList.get(i);
08 %>
09      < tr >
10          < td ><%=onlineUsername%></ td >
11      </ tr >
12 <%
13 }
14 %>
15 </ table >

點擊註銷頁面logout.jsp頁面

01 <%@ page contentType="text/html;charset=utf-8"%>
02 <%@ page import="java.util.*"%>
03 <%
04      // 取得登錄的用戶名
05      String username = (String) session.getAttribute("username");
06      // 銷燬session
07      session.invalidate();
08      // 從在線列表中刪除用戶名
09      List onlineUserList = (List) application.getAttribute("onlineUserList");
10      onlineUserList.remove(username);
11      // 成功
12      response.sendRedirect("index.jsp");
13 %>

OK,登陸、查看、註銷頁面都有了,下面開始新建監聽器

1、HttpSessionListener

添加類OnlineUserListener,繼承HttpSessionListener,HttpSessionListener中有兩個方法 sessionCreated(HttpSessionEvent event)與sessionDestroyed(HttpSessionEvent event),前者是監聽session的新建,後者是監聽session的銷燬。

 OnlineUserListener代碼如下:

 

01 package com.test;
02  
03 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

 

2、HttpSessionBindingListener

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的銷燬。

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