下午比較閒(其實今天都很閒),想了一下在線人數統計方面的實現,上網找了下這方面的知識,最初我的想法是,管理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的銷毀。
作者“ybhanxiao”