還記得我們之前講過的在線列表嗎?第 4.2 節(jié) “例子:在線列表”。我們曾經說過那個在線列表無法判斷用戶非法退出,很可能造成在線列表無限增大,現在我們可以用listener來彌補這一問題了。
如果你不滿足以下任一條件,請繼續(xù)閱讀,否則請?zhí)^此后的部分,進入下一章:第 9 章 封裝taglib組件。
了解如何使用HttpSessionListener監(jiān)聽session的銷毀。
了解如何使用HttpSessionBindingListener監(jiān)聽session的銷毀。
編寫一個OnlineUserListener。
package anni;import java.util.List;import javax.servlet.ServletContext;import javax.servlet.http.HttpSession;import javax.servlet.http.HttpSessionListener;import javax.servlet.http.HttpSessionEvent;public class OnlineUserListener implements HttpSessionListener {public void sessionCreated(HttpSessionEvent event) {}public void sessionDestroyed(HttpSessionEvent event) {HttpSession session = event.getSession();ServletContext application = session.getServletContext();// 取得登錄的用戶名String username = (String) session.getAttribute("username");// 從在線列表中刪除用戶名List onlineUserList = (List) application.getAttribute("onlineUserList");onlineUserList.remove(username);System.out.println(username + "超時退出。");}}
OnlineUserListener實現了HttpSessionListener定義的兩個方法:sessionCreated()和sessionDestroyed()。這兩個方法可以監(jiān)聽到當前應用中session的創(chuàng)建和銷毀情況。我們這里只用到sessionDestroyed()在session銷毀時進行操作就可以。
從HttpSessionEvent中獲得即將銷毀的session,得到session中的用戶名,并從在線列表中刪除。最后一句向console打印一條信息,提示操作成功,這只是為了調試用,正常運行時刪除即可。
為了讓監(jiān)聽器發(fā)揮作用,我們將它添加到web.xml中:
<listener><listener-class>anni.OnlineUserListener</listener-class></listener>
以下兩種情況下就會發(fā)生sessionDestoryed(會話銷毀)事件:
執(zhí)行session.invalidate()方法時。
既然LogoutServlet.java中執(zhí)行session.invalidate()時,會觸發(fā)sessionDestory()從在線用戶列表中清除當前用戶,我們就不必在LogoutServlet.java中對在線列表進行操作了,所以LogoutServlet.java的內容現在是這樣。
public void doGet(HttpServletRequest request,HttpServletResponse response) throws ServletException, IOException { // 銷毀session request.getSession().invalidate(); // 成功 response.sendRedirect("index.jsp"); }
如果用戶長時間沒有訪問服務器,超過了會話最大超時時間,服務器就會自動銷毀超時的session。
會話超時時間可以在web.xml中進行設置,為了容易看到超時效果,我們將超時時間設置為最小值。
<session-config> <session-timeout>1</session-timeout> </session-config>
時間單位是一分鐘,并且只能是整數,如果是零或負數,那么會話就永遠不會超時。
對應例子在08-01,為了驗證OnlineUserListener是否能正常執(zhí)行,我們可以登錄兩個用戶,其中一個點擊注銷,另一個等待一分鐘,然后可以在console中看到輸出的信息。
HttpSessionBindingListener雖然叫做監(jiān)聽器,但使用方法與HttpSessionListener完全不同。我們實際看一下它是如何使用的。
我們的OnlineUserBindingListener實現了HttpSessionBindingListener接口,接口中共定義了兩個方法:valueBound()和valueUnbound(),分別對應數據綁定,和取消綁定兩個事件。
所謂對session進行數據綁定,就是調用session.setAttribute()把HttpSessionBindingListener保存進session中。我們在LoginServlet.java中進行這一步。
// 把用戶名放入在線列表session.setAttribute("onlineUserBindingListener", new OnlineUserBindingListener(username));
這就是HttpSessionBindingListener和HttpSessionListener之間的最大區(qū)別:HttpSessionListener只需要設置到web.xml中就可以監(jiān)聽整個應用中的所有session。HttpSessionBindingListener必須實例化后放入某一個session中,才可以進行監(jiān)聽。
從監(jiān)聽范圍上比較,HttpSessionListener設置一次就可以監(jiān)聽所有session,HttpSessionBindingListener通常都是一對一的。
正是這種區(qū)別成就了HttpSessionBindingListener的優(yōu)勢,我們可以讓每個listener對應一個username,這樣就不需要每次再去session中讀取username,進一步可以將所有操作在線列表的代碼都移入listener,更容易維護。
valueBound()方法的代碼如下:
public void valueBound(HttpSessionBindingEvent event) {HttpSession session = event.getSession();ServletContext application = session.getServletContext();// 把用戶名放入在線列表List onlineUserList = (List) application.getAttribute("onlineUserList");// 第一次使用前,需要初始化if (onlineUserList == null) {onlineUserList = new ArrayList();application.setAttribute("onlineUserList", onlineUserList);}onlineUserList.add(this.username);}
username已經通過構造方法傳遞給listener,在數據綁定時,可以直接把它放入用戶列表。
與之對應的valueUnbound()方法,代碼如下:
public void valueUnbound(HttpSessionBindingEvent event) {HttpSession session = event.getSession();ServletContext application = session.getServletContext();// 從在線列表中刪除用戶名List onlineUserList = (List) application.getAttribute("onlineUserList");onlineUserList.remove(this.username);System.out.println(this.username + "退出。");}
這里可以直接使用listener的username操作在線列表,不必再去擔心session中是否存在username。
valueUnbound的觸發(fā)條件是以下三種情況:
執(zhí)行session.invalidate()時。
session超時,自動銷毀時。
執(zhí)行session.setAttribute("onlineUserListener", "其他對象");或session.removeAttribute("onlineUserListener");將listener從session中刪除時。
因此,只要不將listener從session中刪除,就可以監(jiān)聽到session的銷毀。
例子在08-02目錄下,可以與08-01對比一下異同,兩者的最終效果是相同的。