Servlet中Listener的应用

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

  我们看具体的代码,见示例14-9。

  【程序源代码】

  

      1 // ==================== Program Discription =====================
  2 // 程序名称:示例14-9 : EncodingFilter .java

  3 // 程序目的:学习使用监听器

  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 //创建一个session时激发

  19 public void sessionCreated(HttpSessionEvent se)

  20 {

  21 count++;

  22 setContext(se);

  23

  24 }

  25 //当一个session失效时激发

  26 public void sessionDestroyed(HttpSessionEvent se)

  27 {

  28 count--;

  29 setContext(se);

  30 }

  31 //设置context的属性,它将激发attributeReplaced或attributeAdded方法

  32 public void setContext(HttpSessionEvent se)

  33 {

  34 se.getSession().getServletContext().

  setAttribute("onLine",new Integer(count));

  35 }

  36 //增加一个新的属性时激发

  37 public void attributeAdded(ServletContextAttributeEvent event) {

  38

  39 log("attributeAdded('" + event.getName() + "', '" +

  40 event.getValue() + "')");

  41

  42 }

  43

  44 //删除一个新的属性时激发

  45 public void attributeRemoved(ServletContextAttributeEvent event) {

  46

  47 log("attributeRemoved('" + event.getName() + "', '" +

  48 event.getValue() + "')");

  49

  50 }

  51

  52 //属性被替代时激发

  53 public void attributeReplaced(ServletContextAttributeEvent event) {

  54

  55 log("attributeReplaced('" + event.getName() + "', '" +

  56 event.getValue() + "')");

  57 }

  58 //context删除时激发

  59 public void contextDestroyed(ServletContextEvent event) {

  60

  61 log("contextDestroyed()");

  62 this.context = null;

  63

  64 }

  65

  66 //context初始化时激发

  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代表当前在线的人数,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:

  对一个应用进行全局监听.随应用启动而启动,随应用消失而消失主要有两个方法:

  contextDestroyed(ServletContextEvent event)

  在应用关闭的时候调用

  contextInitialized(ServletContextEvent event)

  在应用启动的时候调用

  这个监听器主要用于一些随着应用启动而要完成的工作,也就是很多人说的我想在容器

  启动的时候干..........

  一般来说对"全局变量"初始化,如

         public void contextInitialized(ServletContextEvent event){
  ServletContex sc = event.getServletContext();

  sc.setAttribute(name,value);

  }
 
  以后你就可以在任何servlet中getServletContext().getAttribute(name);

  我最喜欢用它来做守护性工作,就是在contextInitialized(ServletContextEvent event)

  方法中实现一个Timer,然后就让应用在每次启动的时候让这个Timer工作:

  程序代码:

         public void contextInitialized(ServletContextEvent event){
  timer = new Timer();

  timer.schedule(new TimerTask(){

  public void run(){

  //do any things

  }

  },0,时间间隔);

  }
 
  有人说Timer只能规定从现在开始的多长时间后,每隔多久做一次事或在什么时间做

  一次事,那我想在每月1号或每天12点做一项工作如何做呢?

你只要设一个间隔,然后每次判断一下当时是不是那个时间段就行了啊,比如每月一号做,那你

  时间间隔设为天,即24小时一个循环,然后在run方法中判断当时日期new Date().getDate()==1

  就行了啊.如果是每天的12点,那你时间间隔设为小时,然后在run中判断new Date().getHour()

  ==12,再做某事就行了.

  ServletContextAttributeListener:

  这个监听器主要监听ServletContex对象在setAttribute()和removeAttribute()的事件,注意

  也就是一个"全局变量"在被Add(第一次set),replace(对已有的变量重新赋值)和remove的时候.

  分别调用下面三个方法:

  public void attributeAdded(ServletContextAttributeEvent scab)这个方法不仅可以知道

  哪些全局变量被加进来,而且可获取容器在启动时自动设置了哪些context变量:

  程序代码:

      public void attributeAdded(ServletContextAttributeEvent scab){
  System.out.println(scab.getName());

  }

  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方法,就会自动关闭自己.

  而其它的需要改变状态的对象了是一样.

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值