写点什么

Java 进阶 (十三)servlet 监听器

  • 2022 年 9 月 10 日
    安徽
  • 本文字数:3732 字

    阅读完需:约 12 分钟

Java进阶(十三)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方法却会按一定的策略很久以后才会发生.


HttpSessionAttributeListenerServletContextAttributeListener一样,它监听一个 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 方法,就会自动关闭自己.而其它的需要改变状态的对象了是一样.

发布于: 刚刚阅读数: 4
用户头像

No Silver Bullet 2021.07.09 加入

岂曰无衣 与子同袍

评论

发布
暂无评论
Java进阶(十三)servlet监听器_Java_No Silver Bullet_InfoQ写作社区