Java Web 开发时,可以使用 Listener 来监听来监听一些事件,从而实现一些功能。实际上这个监听器,原理就是 Delphi 中大家常用的各种事件。
1. 那么,监听器的主要用途用哪些呢:
- 统计在线人数和在线用户
- 系统启动时加载初始化信息: 包括各种缓存、共公的定制器、数据库链接等等
- 统计网站访问量
- 路Spring结合
2. 监听器可以按监听的对象来分类:
- ServletContext (ServletContextListener):用于监听应用程序环境对象的事件监听器(一个项目中只有一个),可以用来启动定时器、初始化全局对象。
- HttpSession (HttpSessionListener): 用于监听用户会话对象的事件监听器。
- ServletRequest (ServletRequestListener): 用于监听请求消息对象的事件监听器,可以用来读取参数,记录访问历史等。
3. 按监听的事件来划分:
- 监听域对象自身的创建和销毁的事件监听器(2中的三类)。
- 监听域对象中的属性的增加和删除的事件监听器: ServletContextAttributeListener、HttpSessionAttributeListener、ServletRequestAttributeListener。
- 监听绑定到 HttpSession 域中的某个对象的状态的事件监听器。它分为这几种状态:
- 绑定: 将状态通过 set 保存到 Session 中。
- 解除绑定 : 使用 remove 删除状态。
- 钝化: 将 session 对象持久化到存储设备上。(Session本身是存在于服务器内存中。Session 钝化机制由SessionManager管理: 第一种管理器是 org.apache.catalina.session.StandardManger, 当 tomcat服务器被关闭或重启时,tomcat服务器会将当前内存中的session钝化到服务器文件系统中。另一种情况是web应用程序被重新加载时,内存中的session对象也会被钝化到服务器的文件系统中。钝化后的文件被保存到 Tomcat安装路径下的 /work/Catalina/hostname/applicationname/SESSION.ser 中。第二种管理器是 org.apache.catalina.session.Persistentmanager ,它是在钝化基础上进行了扩张,前两种情况和StandardManager相同,第三种情况,可以配置主流内存的Session对象数目,将不常用的Session对象保存到文件系统或数据库中,当要使用时再进行加载。默认情况下,Tomcat提供两个钝化驱动类:org.apache.Catalina.FileStore 和 org.apache.Catalina.JDBCStore。)
- 活化: 从存储设备上恢复。
绑定、解除绑定使用 HttpSessionBindingListener 接口, 钝化、活化使用 HttpSessionActivationListener 接口。这两个监听器不需要注册。
4. 注册方法
在 Servlet 3.0 之前的版本中, 需要在 web.xml 中进行注册。也比较简单,就是一个声明:
<listener> <listener-class>com.imooc.listener.FirstListener</listener-class> </listener>
在 servlet 3.0 之后, 可以不再到 web.xml 中进行配置, 直接给监听器类加上注解 @WebListener 就可以。
package com.imooc.listener; import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; import javax.servlet.annotation.WebListener; import com.demo.utils.UserManager; @WebListener public class FirstListener implements ServletContextListener { @Override public void contextInitialized(ServletContextEvent sce) { System.out.println("contextInitialized"); } @Override public void contextDestroyed(ServletContextEvent sce) { System.out.println("contextDestroyed"); } }
5. 示例: 使用监听器实现的显示在线用户列表
效果如下:
此示例综合用到了这些技术: jstl和el标签、jsp脚本、listener监听器 、单例全局对象、javabean、线程同步。
主要代码:
index.jsp
<%@ page language="java" contentType="text/html; charset=UTF-8" %> <%@ page import="com.demo.utils.*" %> <%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %> <%@ taglib uri="http://java.sun.com/jsp/jstl/functions" prefix="fn" %> <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>欢迎访问</title> </head> <body> <% request.setAttribute("users", UserManager.getInstance().getItems()); %> <h1>在线用户列表</h1><hr> <!-- 使用 JSTL 输出在线用户列表 --> <table> <tr><th width="80px">name</th><th width="320px">sessionID</th><th width="180px">IP地址</th><th>登录时间</th><tr> <c:forEach var="user" items="${requestScope.users}"> <tr> <td>${user.name}</td> <td>${user.sessionID}</td> <td>${user.ipaddr}</td> <td>${user.firstTimeStr}</td> </tr> </c:forEach></table> <br> 当前在线: <c:out value="${fn:length(users)}"></c:out>人。 </body> </html>
RequestListener.java
package com.demo.listener; import javax.servlet.ServletRequestEvent; import javax.servlet.ServletRequestListener; import javax.servlet.annotation.WebListener; import javax.servlet.http.HttpServletRequest; import com.demo.utils.User; import com.demo.utils.UserManager; @WebListener public class RequestListener implements ServletRequestListener { @Override public void requestDestroyed (ServletRequestEvent sre) { System.out.println("requestDestroyed"); } @Override public void requestInitialized (ServletRequestEvent sre) { HttpServletRequest req = (HttpServletRequest) sre.getServletRequest(); UserManager mgr = UserManager.getInstance(); // 用户请求时,如果 mgr 中不存在当前会放原 sessionID, 则新建一个User对象,加入管理器中 if (!mgr.existSession(req.getSession().getId())) { User item = new User(); item.setFirstTime(System.currentTimeMillis()); item.setIpaddr(req.getRemoteAddr() + ":" + req.getRemotePort()); item.setName(req.getParameter("name")); item.setSessionID(req.getSession().getId()); mgr.addSession(item); } System.out.println("requestInitialized: " + req.getRequestURI() + ", " + req.getParameter("name")); } }
SessionListener.java
package com.demo.listener; import javax.servlet.annotation.WebListener; import javax.servlet.http.HttpSessionEvent; import javax.servlet.http.HttpSessionListener; import com.demo.utils.UserManager; @WebListener public class SessionListener implements HttpSessionListener { @Override public void sessionCreated(HttpSessionEvent se) { System.out.println("sessionCreated"); } @Override public void sessionDestroyed(HttpSessionEvent se) { System.out.println("sessionDestroyed"); // Session 失效时,从列表中删除 UserManager.getInstance().removeSession(se.getSession().getId()); } }
UserManager.java
package com.demo.utils; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; /** 用户管理 */ public class UserManager { private static UserManager Instance; private Map<String, User> items = new HashMap<String, User>(); /** 单例 */ public static UserManager getInstance() { if (Instance == null) Instance = new UserManager(); return Instance; } private UserManager() {} // 由于 Web 请求是并发的,对列表的操作,需要使用 synchronized 关键字线程同步,防止出现异常 public synchronized boolean existSession(String sessionID) { return items.containsKey(sessionID); } public synchronized void addSession(User v) { if (v != null) items.put(v.getSessionID(), v); } public synchronized void removeSession(String sessionID) { if (items.containsKey(sessionID)) items.remove(sessionID); } public int size() { return items.size(); } public synchronized User get(String sessionID) { return items.get(sessionID); } public synchronized List<User> getItems() { ArrayList<User> list = new ArrayList<User>(); for (Map.Entry<String, User> entry : items.entrySet()) list.add(entry.getValue()); return list; } }
User.java
package com.demo.utils; import java.io.Serializable; import java.text.SimpleDateFormat; /** 用户对象 Java Bean */ public class User implements Serializable { private static final long serialVersionUID = 1L; private String name; private String sessionID; private String ipaddr; private long firstTime; public String getName() { return name; } public void setName(String name) { this.name = name; } public String getSessionID() { return sessionID; } public void setSessionID(String sessionID) { this.sessionID = sessionID; } public String getIpaddr() { return ipaddr; } public void setIpaddr(String ipaddr) { this.ipaddr = ipaddr; } public long getFirstTime() { return firstTime; } public void setFirstTime(long firstTime) { this.firstTime = firstTime; } public String getFirstTimeStr() { return new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(firstTime).toString(); } }
完整项目下载:
链接: http://pan.baidu.com/s/1jI2fVqm 密码: wjr5
【感谢】
慕课网、Fcming 老师