Docker 的優點
679
2025-04-01
servlet-
Listener是Servlet的-,它可以監聽客戶端的請求、服務端的操作等。通過-,可以自動激發一些操作,比如監聽在線的用戶的數量。當?增加一個HttpSession時,就激發sessionCreated(HttpSessionEvent?se)方法,這樣就可以給在線人數加1。常用的監聽接口有以下幾個:
ServletContextAttributeListener監聽對ServletContext屬性的操作,比如增加、刪除、修改屬性。
ServletContextListener監聽ServletContext。當創建ServletContext時,激發contextInitialized?(ServletContextEvent?sce)方法;當銷毀ServletContext時,激發contextDestroyed(ServletContextEvent?sce)方法。
HttpSessionListener監聽HttpSession的操作。當創建一個Session時,激發session?Created(HttpSessionEvent?se)方法;當銷毀一個Session時,激發sessionDestroyed?(HttpSessionEvent?se)方法。
HttpSessionAttributeListener監聽HttpSession中的屬性的操作。當在Session增加一個屬性時,激發attributeAdded?(HttpSessionBindingEvent?se)?方法;當在Session刪除一個屬性時,激發attributeRemoved(HttpSessionBindingEvent?se)方法;當在Session屬性被重新設置時,激發attributeReplaced(HttpSessionBindingEvent?se)?方法。
下面我們開發一個具體的例子,這個-能夠統計在線的人數。在ServletContext初始化和銷毀時,在服務器控制臺打印對應的信息。當ServletContext里的屬性增加、改變、刪除時,在服務器控制臺打印對應的信息。
要獲得以上的功能,-必須實現以下3個接口:
HttpSessionListener
ServletContextListener
ServletContextAttributeListener
import javax.servlet.http.*;
import javax.servlet.*;
public class OnLineCountListener implements HttpSessionListener,ServletContextListener,ServletContextAttributeListener
{
private int count;
private ServletContext context = null;
public OnLineCountListener()
{
count=0;
}
//創建一個session時激發
public void sessionCreated(HttpSessionEvent se)
{
count++;
setContext(se);
}
//當一個session失效時激發
public void sessionDestroyed(HttpSessionEvent se)
{
count--;
setContext(se);
}
//設置context的屬性,它將激發attributeReplaced或attributeAdded方法
public void setContext(HttpSessionEvent se)
{
se.getSession().getServletContext().setAttribute("onLine",new Integer(count));
}
//增加一個新的屬性時激發
public void attributeAdded(ServletContextAttributeEvent event) {
log("attributeAdded('" + event.getName() + "', '" +event.getValue() + "')");
}
//刪除一個新的屬性時激發
public void attributeRemoved(ServletContextAttributeEvent event) {
log("attributeRemoved('" + event.getName() + "', '" +event.getValue() + "')");
}
//屬性被替代時激發
public void attributeReplaced(ServletContextAttributeEvent event) {
log("attributeReplaced('" + event.getName() + "', '" +event.getValue() + "')");
}
//context刪除時激發
public void contextDestroyed(ServletContextEvent event) {
log("contextDestroyed()");
this.context = null;
}
//context初始化時激發
public void contextInitialized(ServletContextEvent event) {
this.context = event.getServletContext();
log("contextInitialized()");
}
private void log(String message) {
System.out.println("ContextListener: " + message);
}
}
【程序注解】
在OnLineCountListener?里,用count代表當前在線的人數,OnLineCountListener將在Web服務器啟動時自動執行。當?OnLineCountListener構造好后,把count設置為0。每增加一個Session,OnLineCountListener會自動調用?sessionCreated(HttpSessionEvent?se)方法;每銷毀一個Session,OnLineCountListener會自動調用sessionDestroyed?(HttpSessionEvent?se)方法。當調用sessionCreated(HttpSessionEvent?se)方法時,說明又有一個客戶在請求,此時使在線的人數(count)加1,并且把count寫到ServletContext中。?ServletContext的信息是所有客戶端共享的,這樣,每個客戶端都可以讀取到當前在線的人數。
從作用域范圍來說,Servlet的作用域有ServletContext,HttpSession,ServletRequest.
Context范圍:
ServletContextListener:對一個應用進行全局監聽.隨應用啟動而啟動,隨應用消失而消失主要有兩個方法:
public?void?contextDestroyed(ServletContextEvent?event)?在應用關閉的時候調用
public?void?contextInitialized(ServletContextEvent?event)?在應用啟動的時候調用
這個-主要用于一些隨著應用啟動而要完成的工作,也就是很多人說的我想在容器啟動的時候干..........
一般來說對"全局變量"初始化,如
public?void?contextInitialized(ServletContextEvent?event){
ServletContex?sc?=?event.getServletContext();
sc.setAttribute(name,value);
}
以后你就可以在任何servlet中getServletContext().getAttribute(name);
ServletContextAttributeListener:這個-主要監聽ServletContex對象在?setAttribute()和removeAttribute()的事件,注意也就是一個"全局變量"在被Add(第一次set),replace(對?已有的變量重新賦值)和remove的時候.分別調用下面三個方法:
public?void?attributeAdded(ServletContextAttributeEvent?scab)該方法不僅可以知道哪些全局變量被加進來,而且可獲取容器在啟動時自動設置了哪些context變量.
public?void?attributeRemoved(ServletContextAttributeEvent?scab)
public?void?attributeReplaced(ServletContextAttributeEvent?scab)
Session范圍:
HttpSessionListener:這個-主要監聽一個Session對象被生成和銷毀時發生的事件.對應有兩個方法:
public?void?sessionCreated(HttpSessionEvent?se)
public?void?sessionDestroyed(HttpSessionEvent?se)
一般來說,一個session對象被create時,可以說明有一個新客端進入.可以用來粗略統計在線人數,注意這不是精確的,因為這個客戶端可能立即就關閉了,但sessionDestroyed方法卻會按一定的策略很久以后才會發生.
HttpSessionAttributeListener:和ServletContextAttributeListener一樣,它監聽一個?session對象的Attribut被Add(一個特定名稱的Attribute每一次被設置),replace(已有名稱的Attribute的值被?重設)和remove時的事件.對應有三個方法.
public?void?attributeAdded(HttpSessionBindingEvent?se)
public?void?attributeRemoved(HttpSessionBindingEvent?se)
public?void?attributeReplaced(HttpSessionBindingEvent?se)
上面的幾個-的方法,都是在監聽應用邏輯中servlet邏輯中發生了什么事。一般的來說,我們只要完成邏輯功能,比如?session.setAttribute("aaa","111"),我只要把一個名為aaa的變量放在session中以便以后我能獲取它,并不關心?當session.setAttribute("aaa","111")發生時我還要干什么.(當然有些時候要利用的),但對于下面這個-,你應該好?好發解一下:
HttpSessionBindingListener:
上面的-都是作為一個獨立的Listener在容器中控制事件的.而HttpSessionBindingListener對在一對象中監聽該對象的?狀態,實現了該接口的對象如果被作為value被add到一個session中或從session中remove,它就會知道自己已經作為一個?session對象或已經從session刪除,這對于一些非純JAVA對象,生命周期長于session的對象,以及其它需要釋放資源或改變狀態的對象?非常重要.
比如:
session.setAttribute("abcd","1111");
以后session.removeAttribute("abcd");因為abcd是一個字符中,你從session中remove后,它就會自動被垃?圾回收器回收,而如果是一個connection:(只是舉例,你千萬不要加connection往session中加入)
程序代碼:
session.setAttribute("abcd",conn);
以后session.removeAttribute("abcd");這時這個conn被從session中remove了,你已經無法獲取它的?句柄,所以你根本沒法關閉它.而在沒有remove之前你根本不知道什么時候要被remove,你又無法close(),那么這個connection對?象就死了.另外還有一些對象可以在被加入一個session時要鎖定還要被remove時要解鎖,應因你在程序中無法判斷什么時候被?remove(),add還好操作,我可以先加鎖再add,但remove就后你就找不到它的句柄了,根本沒法解鎖,所以這些操作只能在對象自身中實現.?也就是在對象被add時或remove時通知對象自己回調相應的方法:
程序代碼:
MyConn extends Connection implements HttpSessionBindingListener{
public void valueBound(HttpSessionBindingEvent se){
this.initXXX();
}
public void valueUnbound(HttpSessionBindingEvent se){
this.close();
}
}
session.setAttribute("aaa",new MyConn());
這時如果調用session.removeAttribute("aaa"),則觸發valueUnbound方法,就會自動關閉自己.而其它的需要改變狀態的對象了是一樣.
Java Servlet 容器
版權聲明:本文內容由網絡用戶投稿,版權歸原作者所有,本站不擁有其著作權,亦不承擔相應法律責任。如果您發現本站中有涉嫌抄襲或描述失實的內容,請聯系我們jiasou666@gmail.com 處理,核實后本網站將在24小時內刪除侵權內容。
版權聲明:本文內容由網絡用戶投稿,版權歸原作者所有,本站不擁有其著作權,亦不承擔相應法律責任。如果您發現本站中有涉嫌抄襲或描述失實的內容,請聯系我們jiasou666@gmail.com 處理,核實后本網站將在24小時內刪除侵權內容。