捕獲Session事件的意義:
1、 記錄網(wǎng)站的客戶登錄日志(登錄,退出信息等)
2、 統(tǒng)計(jì)在線人數(shù)
3、 等等還有很多,呵呵,自己想吧……總之挺重要的。
Session代表客戶的會(huì)話過程,客戶登錄時(shí),往Session中傳入一個(gè)對(duì)象,即可跟蹤客戶的會(huì)話。在Servlet中,傳入Session的對(duì)象如果是一個(gè)實(shí)現(xiàn)HttpSessionBindingListener接口的對(duì)象(方便起見,此對(duì)象稱為監(jiān)聽器),則在傳入的時(shí)候(即調(diào)用HttpSession對(duì)象的setAttribute方法的時(shí)候)和移去的時(shí)候(即調(diào)用HttpSession對(duì)象的removeAttribute方法的時(shí)候或Session Time out的時(shí)候)Session對(duì)象會(huì)自動(dòng)調(diào)用監(jiān)聽器的valueBound和valueUnbound方法(這是HttpSessionBindingListener接口中的方法)。由此可知,登錄日志也就不難實(shí)現(xiàn)了。
另外一個(gè)問題是,如何統(tǒng)計(jì)在線人數(shù),這個(gè)問題跟實(shí)現(xiàn)登錄日志稍微有點(diǎn)不同,統(tǒng)計(jì)在線人數(shù)(及其信息),就是統(tǒng)計(jì)現(xiàn)在有多少個(gè)Session實(shí)例存在,我們可以增加一個(gè)計(jì)數(shù)器(如果想存儲(chǔ)更多的信息,可以用一個(gè)對(duì)象來做計(jì)數(shù)器,隨后給出的實(shí)例中,簡(jiǎn)單起見,用一個(gè)整數(shù)變量作為計(jì)數(shù)器),通過在valueBound方法中給計(jì)數(shù)器加1,valueUnbound方法中計(jì)數(shù)器減1,即可實(shí)現(xiàn)在線人數(shù)的統(tǒng)計(jì)。當(dāng)然,這里面要利用到ServletContext的全局特性。(有關(guān)ServletContext的敘述請(qǐng)參考Servlet規(guī)范),新建一個(gè)監(jiān)聽器,并將其實(shí)例存入ServletContext的屬性中,以保證此監(jiān)聽器實(shí)例的唯一性,當(dāng)客戶登錄時(shí),先判斷ServletContext的這個(gè)屬性是否為空,如果不為空,證明已經(jīng)創(chuàng)建,直接將此屬性取出放入Session中,計(jì)數(shù)器加1;如果為空則創(chuàng)建一個(gè)新的監(jiān)聽器,并存入ServletContext的屬性中。
舉例說明:
實(shí)現(xiàn)一個(gè)監(jiān)聽器:
// SessionListener.java
import java.io.*;
import java.util.*;
import javax.servlet.http.*;
//監(jiān)聽登錄的整個(gè)過程
public class SessionListener implements HttpSessionBindingListener
{
public String privateInfo=""; //生成監(jiān)聽器的初始化參數(shù)字符串
private String logString=""; //日志記錄字符串
private int count=0; //登錄人數(shù)計(jì)數(shù)器
public SessionListener(String info){
this.privateInfo=info;
}
public int getCount(){
return count;
}
public void valueBound(HttpSessionBindingEvent event)
{
count++;
if (privateInfo.equals("count"))
{
return;
}
try{
Calendar calendar=new GregorianCalendar();
System.out.println("LOGIN:"+privateInfo+" TIME:"+calendar.getTime());
logString="\nLOGIN:"+privateInfo+" TIME:"+calendar.getTime()+"\n";
for(int i=1;i<1000;i++){
File file=new File("yeeyoo.log"+i);
if(!(file.exists()))
file.createNewFile(); //如果文件不存在,創(chuàng)建此文件
if(file.length()>1048576) //如果文件大于1M,重新創(chuàng)建一個(gè)文件
continue;
FileOutputStream foo=new FileOutputStream("yeeyoo.log"+i,true);
//以append方式打開創(chuàng)建文件
foo.write(logString.getBytes(),0,logString.length()); //寫入日志字符串
foo.close();
break;//退出
}
}catch(FileNotFoundException e){}
catch(IOException e){}
}
public void valueUnbound(HttpSessionBindingEvent event)
{
count--;
if (privateInfo.equals("count"))
{
return;
}
try{
Calendar calendar=new GregorianCalendar();
System.out.println("LOGOUT:"+privateInfo+" TIME:"+calendar.getTime());
logString="\nLOGOUT:"+privateInfo+" TIME:"+calendar.getTime()+"\n";
for(int i=1;i<1000;i++){
File file=new File("yeeyoo.log"+i);
if(!(file.exists()))
file.createNewFile(); //如果文件不存在,創(chuàng)建此文件
if(file.length()>1048576) //如果文件大于1M,重新創(chuàng)建一個(gè)文件
continue;
FileOutputStream foo=new FileOutputStream("yeeyoo.log"+i,true);
//以append方式打開創(chuàng)建文件
foo.write(logString.getBytes(),0,logString.length()); //寫入日志字符串
foo.close();
break;//退出
}
}catch(FileNotFoundException e){}
catch(IOException e){}
}
}
登錄日志的實(shí)現(xiàn):
下面再來看看我們的登錄Servlet中使用這個(gè)監(jiān)聽器的部分源代碼:
……
HttpSession session = req.getSession (true);
……
SessionListener sessionListener=
new SessionListener("IP:"+req.getRemoteAddr());
//對(duì)于每一個(gè)會(huì)話過程均啟動(dòng)一個(gè)監(jiān)聽器
session.setAttribute("listener",sessionListener);
//將監(jiān)聽器植入HttpSession,這將激發(fā)監(jiān)聽器調(diào)用valueBound方法,
//從而記錄日志文件?! ?
當(dāng)系統(tǒng)退出登錄時(shí),只需簡(jiǎn)單地調(diào)用session.removeAttribute(“l(fā)istener”);
即可自動(dòng)調(diào)用監(jiān)聽器的valueUnbound方法?;蛘?,當(dāng)Session Time Out的時(shí)候也會(huì)調(diào)用此方法。
登錄人數(shù)的統(tǒng)計(jì):
ServletContext session1=getServletConfig().getServletContext();
//取得ServletContext對(duì)象實(shí)例
if((SessionListener)session1.getAttribute("listener1")==null)
{
SessionListener sessionListener1=new SessionListener("count");
//只設(shè)置一次,不同于上面日志文件的記錄每次會(huì)話均設(shè)置。
//即當(dāng)?shù)谝粋€(gè)客戶連接到服務(wù)器時(shí)啟動(dòng)一個(gè)全局變量,
//此后所有的客戶將使用相同的上下文。
session1.setAttribute("listener1",sessionListener1);
//將監(jiān)聽器對(duì)象設(shè)置成ServletContext的屬性,具有全局范圍有效性,
//即所有的客戶均可以取得它的實(shí)例。
}
session.setAttribute("listener1",(SessionListener)session1.
getAttribute("listener1"));
//取出此全局對(duì)象,并且將此對(duì)象綁定到某個(gè)會(huì)話中,
//此舉將促使監(jiān)聽器調(diào)用valueBound,計(jì)數(shù)器加一。
在此后的程序中隨時(shí)可以用以下代碼取得當(dāng)前的登錄人數(shù):
((SessionListener)session.getAttribute("listener1")).getCount()
getCount()是監(jiān)聽器的一個(gè)方法,即取得當(dāng)前計(jì)數(shù)器的值也就是登錄人數(shù)了。