能夠監聽(tīng)到session,application的create,destroy,可以監聽(tīng)到session,application
屬性綁定的變化,考慮了一下,可以應用在"在線(xiàn)人數統計","數據緩存"等各個(gè)方面,
下面是整理的一些資料.
Listener是Servlet的監聽(tīng)器,它可以監聽(tīng)客戶(hù)端的請求、服務(wù)端的操作等。通過(guò)監聽(tīng)器,可以自動(dòng)激發(fā)一些操作,比如監聽(tīng)在線(xiàn)的用戶(hù)的數量。當增加一個(gè)HttpSession時(shí),就激發(fā)sessionCreated(HttpSessionEventse)方法,這樣就可以給在線(xiàn)人數加1。常用的監聽(tīng)接口有以下幾個(gè):
ServletContextAttributeListener監聽(tīng)對ServletContext屬性的操作,比如增加、刪除、修改屬性。
ServletContextListener監聽(tīng)ServletContext。當創(chuàng )建ServletContext時(shí),激發(fā)contextInitialized(ServletContextEventsce)方法;當銷(xiāo)毀ServletContext時(shí),激發(fā)contextDestroyed(ServletContextEvent sce)方法。
HttpSessionListener監聽(tīng)HttpSession的操作。當創(chuàng )建一個(gè)Session時(shí),激發(fā)session Created(HttpSessionEventse)方法;當銷(xiāo)毀一個(gè)Session時(shí),激發(fā)sessionDestroyed (HttpSessionEvent se)方法。
HttpSessionAttributeListener監聽(tīng)HttpSession中的屬性的操作。當在Session增加一個(gè)屬性時(shí),激發(fā)attributeAdded(HttpSessionBindingEvent se)方法;當在Session刪除一個(gè)屬性時(shí),激發(fā)attributeRemoved(HttpSessionBindingEventse)方法;當在Session屬性被重新設置時(shí),激發(fā)attributeReplaced(HttpSessionBindingEvent se)方法。
下面我們開(kāi)發(fā)一個(gè)具體的例子,這個(gè)監聽(tīng)器能夠統計在線(xiàn)的人數。在ServletContext初始化和銷(xiāo)毀時(shí),在服務(wù)器控制臺打印對應的信息。當ServletContext里的屬性增加、改變、刪除時(shí),在服務(wù)器控制臺打印對應的信息。
要獲得以上的功能,監聽(tīng)器必須實(shí)現以下3個(gè)接口:
HttpSessionListener
ServletContextListener
ServletContextAttributeListener
我們看具體的代碼,見(jiàn)示例14-9。
【程序源代碼】
1 // ==================== Program Discription =====================
2 // 程序名稱(chēng):示例14-9 : EncodingFilter .java
3 // 程序目的:學(xué)習使用監聽(tīng)器
4 // ==============================================================
5 import javax.servlet.http.*;
6 import javax.servlet.*;
7
8 public class OnLineCountListener implements HttpSessionListener,
ServletContextListener,ServletContextAttributeListener
9 {
10 private int count;
11 private ServletContext context = null;
12
13 public OnLineCountListener()
14 {
15 count=0;
16 //setContext();
17 }
18 //創(chuàng )建一個(gè)session時(shí)激發(fā)
19 public void sessionCreated(HttpSessionEvent se)
20 {
21 count++;
22 setContext(se);
23
24 }
25 //當一個(gè)session失效時(shí)激發(fā)
26 public void sessionDestroyed(HttpSessionEvent se)
27 {
28 count--;
29 setContext(se);
30 }
31 //設置context的屬性,它將激發(fā)attributeReplaced或attributeAdded方法
32 public void setContext(HttpSessionEvent se)
33 {
34 se.getSession().getServletContext().
setAttribute("onLine",new Integer(count));
35 }
36 //增加一個(gè)新的屬性時(shí)激發(fā)
37 public void attributeAdded(ServletContextAttributeEvent event) {
38
39 log("attributeAdded(‘" + event.getName() + "‘, ‘" +
40 event.getValue() + "‘)");
41
42 }
43
44 //刪除一個(gè)新的屬性時(shí)激發(fā)
45 public void attributeRemoved(ServletContextAttributeEvent event) {
46
47 log("attributeRemoved(‘" + event.getName() + "‘, ‘" +
48 event.getValue() + "‘)");
49
50 }
51
52 //屬性被替代時(shí)激發(fā)
53 public void attributeReplaced(ServletContextAttributeEvent event) {
54
55 log("attributeReplaced(‘" + event.getName() + "‘, ‘" +
56 event.getValue() + "‘)");
57 }
58 //context刪除時(shí)激發(fā)
59 public void contextDestroyed(ServletContextEvent event) {
60
61 log("contextDestroyed()");
62 this.context = null;
63
64 }
65
66 //context初始化時(shí)激發(fā)
67 public void contextInitialized(ServletContextEvent event) {
68
69 this.context = event.getServletContext();
70 log("contextInitialized()");
71
72 }
73 private void log(String message) {
74
75 System.out.println("ContextListener: " + message);
76 }
77 }
【程序注解】
在OnLineCountListener里,用count代表當前在線(xiàn)的人數,OnLineCountListener將在Web服務(wù)器啟動(dòng)時(shí)自動(dòng)執行。當OnLineCountListener構造好后,把count設置為0。每增加一個(gè)Session,OnLineCountListener會(huì )自動(dòng)調用sessionCreated(HttpSessionEventse)方法;每銷(xiāo)毀一個(gè)Session,OnLineCountListener會(huì )自動(dòng)調用sessionDestroyed(HttpSessionEvent se)方法。當調用sessionCreated(HttpSessionEventse)方法時(shí),說(shuō)明又有一個(gè)客戶(hù)在請求,此時(shí)使在線(xiàn)的人數(count)加1,并且把count寫(xiě)到ServletContext中。ServletContext的信息是所有客戶(hù)端共享的,這樣,每個(gè)客戶(hù)端都可以讀取到當前在線(xiàn)的人數。
從作用域范圍來(lái)說(shuō),Servlet的作用域有ServletContext,HttpSession,ServletRequest.
Context范圍:
ServletContextListener:
對一個(gè)應用進(jìn)行全局監聽(tīng).隨應用啟動(dòng)而啟動(dòng),隨應用消失而消失主要有兩個(gè)方法:
contextDestroyed(ServletContextEvent event)
在應用關(guān)閉的時(shí)候調用
contextInitialized(ServletContextEvent event)
在應用啟動(dòng)的時(shí)候調用
這個(gè)監聽(tīng)器主要用于一些隨著(zhù)應用啟動(dòng)而要完成的工作,也就是很多人說(shuō)的我想在容器
啟動(dòng)的時(shí)候干..........
一般來(lái)說(shuō)對"全局變量"初始化,如
public void contextInitialized(ServletContextEvent event){
ServletContex sc = event.getServletContext();
sc.setAttribute(name,value);
}
以后你就可以在任何servlet中g(shù)etServletContext().getAttribute(name);
我最喜歡用它來(lái)做守護性工作,就是在contextInitialized(ServletContextEvent event)
方法中實(shí)現一個(gè)Timer,然后就讓?xiě)迷诿看螁?dòng)的時(shí)候讓這個(gè)Timer工作:
程序代碼:
public void contextInitialized(ServletContextEvent event){
timer = new Timer();
timer.schedule(new TimerTask(){
public void run(){
//do any things
}
},0,時(shí)間間隔);
}
有人說(shuō)Timer只能規定從現在開(kāi)始的多長(cháng)時(shí)間后,每隔多久做一次事或在什么時(shí)間做
一次事,那我想在每月1號或每天12點(diǎn)做一項工作如何做呢?
你只要設一個(gè)間隔,然后每次判斷一下當時(shí)是不是那個(gè)時(shí)間段就行了啊,比如每月一號做,那你
時(shí)間間隔設為天,即24小時(shí)一個(gè)循環(huán),然后在run方法中判斷當時(shí)日期new Date().getDate()==1
就行了啊.如果是每天的12點(diǎn),那你時(shí)間間隔設為小時(shí),然后在run中判斷new Date().getHour()
==12,再做某事就行了.
ServletContextAttributeListener:
這個(gè)監聽(tīng)器主要監聽(tīng)ServletContex對象在setAttribute()和removeAttribute()的事件,注意
也就是一個(gè)"全局變量"在被Add(第一次set),replace(對已有的變量重新賦值)和remove的時(shí)候.
分別調用下面三個(gè)方法:
public void attributeAdded(ServletContextAttributeEvent scab)這個(gè)方法不僅可以知道
哪些全局變量被加進(jìn)來(lái),而且可獲取容器在啟動(dòng)時(shí)自動(dòng)設置了哪些context變量:
程序代碼:
public void attributeAdded(ServletContextAttributeEvent scab){
System.out.println(scab.getName());
}
public void attributeRemoved(ServletContextAttributeEvent scab)
public void attributeReplaced(ServletContextAttributeEvent scab)
Session范圍:
HttpSessionListener:
這個(gè)監聽(tīng)器主要監聽(tīng)一個(gè)Session對象被生成和銷(xiāo)毀時(shí)發(fā)生的事件.對應有兩個(gè)方法:
程序代碼:
public void sessionCreated(HttpSessionEvent se)
public void sessionDestroyed(HttpSessionEvent se)
一般來(lái)說(shuō),一個(gè)session對象被create時(shí),可以說(shuō)明有一個(gè)新客端進(jìn)入.可以用來(lái)粗略統計在線(xiàn)人
數,注意這不是精確的,因為這個(gè)客戶(hù)端可能立即就關(guān)閉了,但sessionDestroyed方法卻會(huì )按一定
的策略很久以后才會(huì )發(fā)生.
HttpSessionAttributeListener:
和ServletContextAttributeListener一樣,它監聽(tīng)一個(gè)session對象的Attribut被Add(一個(gè)特定
名稱(chēng)的Attribute每一次被設置),replace(已有名稱(chēng)的Attribute的值被重設)和remove時(shí)的事件.
對就的有三個(gè)方法.
程序代碼:
public void attributeAdded(HttpSessionBindingEvent se)
public void attributeRemoved(HttpSessionBindingEvent se)
public void attributeReplaced(HttpSessionBindingEvent se)
上面的幾個(gè)監聽(tīng)器的方法,都是在監聽(tīng)應用邏輯中servlet邏輯中發(fā)生了什么事,一般的來(lái)說(shuō).
我們只要完成邏輯功能,比如session.setAttribute("aaa","111");我只要把一個(gè)名為aaa的變量
放在session中以便以后我能獲取它,我并不關(guān)心當session.setAttribute("aaa","111");發(fā)生時(shí)
我還要干什么.(當然有些時(shí)候要利用的),但對于下面這個(gè)監聽(tīng)器,你應該好好發(fā)解一下:
HttpSessionBindingListener:
上面的監聽(tīng)器都是作為一個(gè)獨立的Listener在容器中控制事件的.而HttpSessionBindingListener
對在一對象中監聽(tīng)該對象的狀態(tài),實(shí)現了該接口的對象如果被作為value被add到一個(gè)session中或從
session中remove,它就會(huì )知道自己已經(jīng)作為一個(gè)session對象或已經(jīng)從session刪除,這對于一些非
純JAVA對象,生命周期長(cháng)于session的對象,以及其它需要釋放資源或改變狀態(tài)的對象非常重要.
比如:
session.setAttribute("abcd","1111");
以后session.removeAttribute("abcd");因為abcd是一個(gè)字符中,你從session中remove后,它就會(huì )
自動(dòng)被垃圾回收器回收,而如果是一個(gè)connection:(只是舉例,你千萬(wàn)不要加connection往session
中加入)
程序代碼:
session.setAttribute("abcd",conn);
以后session.removeAttribute("abcd");這時(shí)這個(gè)conn被從session中remove了,你已經(jīng)無(wú)法獲取它
的句柄,所以你根本沒(méi)法關(guān)閉它.而在沒(méi)有remove之前你根本不知道什么時(shí)候要被remove,你又無(wú)法
close(),那么這個(gè)connection對象就死了.另外還有一些對象可以在被加入一個(gè)session時(shí)要鎖定
還要被remove時(shí)要解鎖,應因你在程序中無(wú)法判斷什么時(shí)候被remove(),add還好操作,我可以先加鎖
再add,但remove就后你就找不到它的句柄了,根本沒(méi)法解鎖,所以這些操作只能在對象自身中實(shí)現.
也就是在對象被add時(shí)或remove時(shí)通知對象自己回調相應的方法:
程序代碼:
MyConn extends Connection implements HttpSessionBindingListener{
public void valueBound(HttpSessionBindingEvent se){
this.initXXX();
}
public void valueUnbound(HttpSessionBindingEvent se){
this.close();
}
}
聯(lián)系客服