在線人數統計方面的實現,上網找了下這方面的知識,最初我的想法是,
管理session,如果session銷燬了就減少,如果登陸用戶了就新增一個,但是如果是用戶非法退出,如:未註銷,關閉瀏覽器等,這個用戶的
session是管理不到的,最後決定用HttpSessionListener接口或HttpSessionBindingListener接口來實
現,通過監聽session的新建和銷燬來控制,詳細如下。
先添加登陸的頁面index.jsp
01
|
<%@ page contentType="text/html;charset=utf-8"%>
|
04
|
<
title
>test</
title
>
|
07
|
<
form
action
=
"login.jsp"
method
=
"post"
>
|
08
|
用戶名:<
input
type
=
"text"
name
=
"username"
/>
|
10
|
<
input
type
=
"submit"
value
=
"登錄"
/>
|
點擊登陸後跳轉的login.jsp(爲了方便,用jsp做servlet,同學們用的時候記得改過來)
01
|
<%@ page contentType="text/html;charset=utf-8"%>
|
02
|
<%@ page import="java.util.*"%>
|
04
|
request.setCharacterEncoding("UTF-8");
|
06
|
String username = request.getParameter("username");
|
08
|
session.setAttribute("username", username);
|
10
|
List onlineUserList = (List) application.getAttribute("onlineUserList");
|
12
|
if (onlineUserList == null) {
|
13
|
onlineUserList = new ArrayList();
|
14
|
application.setAttribute("onlineUserList", onlineUserList);
|
16
|
onlineUserList.add(username);
|
18
|
response.sendRedirect("result.jsp");
|
登陸成功跳轉到顯示頁面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
>
|
05
|
List onlineUserList = (List) application.getAttribute("onlineUserList");
|
06
|
for (int i = 0; i <
onlineUserList.size
(); i++) {
|
07
|
String onlineUsername = (String) onlineUserList.get(i);
|
10
|
<
td
><%=onlineUsername%></
td
>
|
點擊註銷頁面logout.jsp頁面
01
|
<%@ page contentType="text/html;charset=utf-8"%>
|
02
|
<%@ page import="java.util.*"%>
|
05
|
String username = (String) session.getAttribute("username");
|
09
|
List onlineUserList = (List) application.getAttribute("onlineUserList");
|
10
|
onlineUserList.remove(username);
|
12
|
response.sendRedirect("index.jsp");
|
OK,登陸、查看、註銷頁面都有了,下面開始新建監聽器
1、HttpSessionListener
添加類OnlineUserListener,繼承HttpSessionListener,HttpSessionListener中有兩個方法
sessionCreated(HttpSessionEvent event)與sessionDestroyed(HttpSessionEvent
event),前者是監聽session的新建,後者是監聽session的銷燬。
OnlineUserListener代碼如下:
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;
|
11
|
public
class
OnlineUserListener
implements
HttpSessionListener {
|
13
|
public
void
sessionCreated(HttpSessionEvent event) {
|
14
|
System.out.println(
"新建session:"
+event.getSession().getId());
|
16
|
public
void
sessionDestroyed(HttpSessionEvent event) {
|
17
|
HttpSession session = event.getSession();
|
18
|
ServletContext application = session.getServletContext();
|
20
|
String username = (String) session.getAttribute(
"username"
);
|
22
|
List onlineUserList = (List) application.getAttribute(
"onlineUserList"
);
|
23
|
onlineUserList.remove(username);
|
24
|
System.out.println(username+
"已經退出!"
);
|
web.xml配置:
2
|
<
listener-class
>com.test.OnlineUserListener</
listener-class
>
|
一旦監聽器發現調用了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.*"%>
|
05
|
request.setCharacterEncoding("UTF-8");
|
07
|
String username = request.getParameter("username");
|
09
|
session.setAttribute("onlineUserBindingListener", new OnlineUserBindingListener(username));
|
11
|
response.sendRedirect("result.jsp");
|
這就是HttpSessionBindingListener和HttpSessionListener之間的最大區
別:HttpSessionListener只需要設置到web.xml中就可以監聽整個應用中的所有session。
HttpSessionBindingListener必須實例化後放入某一個session中,纔可以進行監聽。
從監聽範圍上比較,HttpSessionListener設置一次就可以監聽所有session,HttpSessionBindingListener通常都是一對一的。
正是這種區別成就了HttpSessionBindingListener的優勢,我們可以讓每個listener對應一個username,這樣
就不需要每次再去session中讀取username,進一步可以將所有操作在線列表的代碼都移入listener,更容易維護。
HttpSessionBindingListener代碼如下:
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;
|
10
|
public
class
OnlineUserBindingListener
implements
HttpSessionBindingListener {
|
13
|
public
OnlineUserBindingListener(String username){
|
14
|
this
.username=username;
|
16
|
public
void
valueBound(HttpSessionBindingEvent event) {
|
17
|
HttpSession session = event.getSession();
|
18
|
ServletContext application = session.getServletContext();
|
20
|
List onlineUserList = (List) application.getAttribute(
"onlineUserList"
);
|
22
|
if
(onlineUserList ==
null
) {
|
23
|
onlineUserList =
new
ArrayList();
|
24
|
application.setAttribute(
"onlineUserList"
, onlineUserList);
|
26
|
onlineUserList.add(
this
.username);
|
29
|
public
void
valueUnbound(HttpSessionBindingEvent event) {
|
30
|
HttpSession session = event.getSession();
|
31
|
ServletContext application = session.getServletContext();
|
34
|
List onlineUserList = (List) application.getAttribute(
"onlineUserList"
);
|
35
|
onlineUserList.remove(
this
.username);
|
36
|
System.out.println(
this
.username +
"退出。"
);
|
這裏可以直接使用listener的username操作在線列表,不必再去擔心session中是否存在username。
valueUnbound的觸發條件是以下三種情況:
a.執行session.invalidate()時。
b.session超時,自動銷燬時。
c.執行session.setAttribute("onlineUserListener", "其他對象");或session.removeAttribute("onlineUserListener");將listener從session中刪除時。
因此,只要不將listener從session中刪除,就可以監聽到session的銷燬。