java监听器(Listener)学习笔记
现在来说说Servlet的监听器Listener,它是实现了javax.servlet.ServletContextListener 接口的服务器端程序,它也是随web应用的启动
而启动,只初始化一次,随web应用的停止而销毁。主要作用是:做一些初始化的内容添加工作、设置一些基本的内容、比如一些参数或者是一些
固定的对象等等。首先来看一下ServletContextListener接口的源代码:
public abstract interface ServletContextListener extends EventListener{ public abstract void contextInitialized(ServletContextEvent paramServletContextEvent); public abstract void contextDestroyed(ServletContextEvent paramServletContextEvent); }
根据ServletContextListener监听器对数据库连接池DataSource的初始化演示它的使用:ListenerWW.java
package listener; import javax.servlet.ServletContext; import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; /** * 现在来说说Servlet的监听器Listener,它是实现了javax.servlet.ServletContextListener 接口的 * 服务器端程序,它也是随web应用的启动而启动,只初始化一次,随web应用的停止而销毁。主要作用是:做一些初始化 * 的内容添加工作、设置一些基本的内容、比如一些参数或者是一些固定的对象等等。 * 示例代码:使用监听器对数据库连接池DataSource进行初始化 */ public class ListenerWW implements ServletContextListener { //销毁方法 public void contextDestroyed(ServletContextEvent servletContextEvent) { ServletContext servletContext = servletContextEvent.getServletContext(); // 在整个web应用销毁之前调用,将所有应用空间所设置的内容清空 servletContext.removeAttribute("dataSource"); System.out.println("销毁工作完成..."); } //初始化方法 public void contextInitialized(ServletContextEvent servletContextEvent) { // 通过这个事件可以获取整个应用的空间 ServletContext servletContext = servletContextEvent.getServletContext(); // 在整个web应用下面启动的时候做一些初始化的内容添加工作 // 未导入对应jar包 BasicDataSource basicDataSource = new BasicDataSource(); basicDataSource.setDriverClassName("com.jdbc.Driver"); basicDataSource.setUrl("jdbc:mysqlocalhost:3306/"); basicDataSource.setUsername("root"); basicDataSource.setPassword("root"); basicDataSource.setMaxActive(10);//最大连接数 basicDataSource.setMaxIdle(5);//最大管理数 // 把 DataSource 放入ServletContext空间中, // 供整个web应用的使用(获取数据库连接) servletContext.setAttribute("dataSource", basicDataSource); System.out.println("应用监听器初始化工作完成..."); System.out.println("已经创建DataSource..."); } }
配置监听器的web.xml
<!-- 配置监听器 --> <listener> <listener-class>listener.ListenerWW</listener-class> </listener>
监听器在实际项目中的应用,监听器在java web中应用的较多,比如:统计当前在线人数、自定义session扫描器。
--------------------- 应用一:统计当前在线人数 ---------------------
代码实现:
package listener; import javax.servlet.ServletContext; import javax.servlet.http.HttpSessionEvent; import javax.servlet.http.HttpSessionListener; /** * 说说Session的监听器Listener,它是实现了javax.servlet.HttpSessionListener 接口的 * 例子:在线人数统计 * @author 旺旺 */ public class ListenerWW implements HttpSessionListener { //在线总人数 private static int onLinePersonCount = 0; //session创建 public void sessionCreated(HttpSessionEvent sessionEvent) { //获取Servlet上下文 ServletContext servletContext = sessionEvent.getSession().getServletContext(); onLinePersonCount = (Integer) servletContext.getAttribute("onLinePersonCount"); // 如果用户登录,TOTAL_ONLINE_USERS自增1 onLinePersonCount++; servletContext.setAttribute("TOTAL_ONLINE_USERS", onLinePersonCount); } //session销毁 public void sessionDestroyed(HttpSessionEvent sessionEvent) { //获取Servlet上下文 ServletContext servletContext = sessionEvent.getSession().getServletContext(); onLinePersonCount = (Integer) servletContext.getAttribute("onLinePersonCount"); // 如果用户登录,TOTAL_ONLINE_USERS自增1 onLinePersonCount--; servletContext.setAttribute("TOTAL_ONLINE_USERS", onLinePersonCount); } }
再来一个例子,扫描Session,清理长时间没有操作的用户
监听实例:
package listener; import java.util.Collections; import java.util.LinkedList; import java.util.List; import java.util.Timer; import javax.servlet.ServletContext; import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; import javax.servlet.http.HttpSession; import javax.servlet.http.HttpSessionEvent; import javax.servlet.http.HttpSessionListener; /** * 当网站用户量增加时,session占用的内存会越来越大,这时session的管理,将会是一项很大的 * 系统开销,为了高效的管理session,我们可以写一个监听器,定期清理掉过期的session * @author 旺旺 */ public class ListenerWW implements HttpSessionListener,ServletContextListener { // 创建一个线程安全的集合,用来存储session List<HttpSession> sessionList = Collections.synchronizedList(new LinkedList<HttpSession>()); private Object lock = new Object(); //session创建 public void sessionCreated(HttpSessionEvent sessionEvent) { HttpSession httpSession = sessionEvent.getSession(); synchronized (lock){ sessionList.add(httpSession); } } //session销毁 public void sessionDestroyed(HttpSessionEvent sessionEvent) { HttpSession httpSession = sessionEvent.getSession(); synchronized (lock){ sessionList.remove(httpSession); } } // web应用关闭时触发contextDestroyed事件 public void contextDestroyed(ServletContextEvent servletContextEvent) { System.out.println("web应用关闭..."); } // web应用启动时触发contextInitialized事件 public void contextInitialized(ServletContextEvent servletContextEvent) { System.out.println("web应用初始化..."); // 创建定时器 Timer timer = new Timer(); // 每隔30秒就定时执行任务 timer.schedule(new WTask(sessionList,lock), 0, 1000*30); } }
定时任务处理类
package listener; import java.util.List; import java.util.ListIterator; import java.util.TimerTask; import javax.servlet.http.HttpSession; public class WTask extends TimerTask{ private List<HttpSession> list; // 存储传递过来的锁 private Object lock; // 构造方法 WTask(List<HttpSession> list, Object lock){ this.list = list; this.lock = lock; } @Override public void run() { // 考虑到多线程的情况,这里必须要同步 synchronized (lock) { ListIterator<HttpSession> sessionIterator = list.listIterator(); while(sessionIterator.hasNext()) { //五分钟没有访问后台 // httpSession.getLastAccessedTime() = session的最后访问时间 HttpSession httpSession = sessionIterator.next(); if((System.currentTimeMillis() - httpSession.getLastAccessedTime()) > 1000*60*5) { // 手动销毁session httpSession.invalidate(); // 从集合中移除已经被销毁的session sessionIterator.remove(); } } /*for(HttpSession httpSession : list) { //五分钟没有访问后台 // httpSession.getLastAccessedTime() = session的最后访问时间 if((System.currentTimeMillis() - httpSession.getLastAccessedTime()) > 1000*60*5) { // 手动销毁session httpSession.invalidate(); // 从集合中移除已经被销毁的session list.remove(httpSession); } }*/ } } }