然後再配置監聽,
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,這個時間也可以根據實際需要進行設置。
後來發現這種方式經常會出現數據丟失的情況,後來又發現通過實現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.*"%>
|
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的銷燬。