监听器实现案例----自定义session扫描器和统计在线用户人数及用户信息
监听器实现案例----自定义session扫描器和统计在线用户人数及用户信息
一、案例一:自定义Session扫描器
1、案例说明
当一个Web应用创建的Session很多时,为了避免Session占用太多的内存,我们可以选择手动将这些内存中的session销毁,那么此时也可以借助监听器技术来实现。对于拿到 每个session 对象, 判断session的最后一次访问时间 与当前时间 的间隔是否超过 5 分钟, 如果超过就手动销毁
2、实现代码
SessionScanner:session对象的监听器
MyTimerTask:定时器timer的任务对象
SessionScanner监听器,使用servlet3.0新特性,使用注解@WebListener完成注册
(1)SessionScanner类
package sessionScanner; import java.util.Collections; import java.util.LinkedList; import java.util.List; import java.util.Timer; import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; import javax.servlet.annotation.WebListener; import javax.servlet.http.HttpSession; import javax.servlet.http.HttpSessionEvent; import javax.servlet.http.HttpSessionListener; /* * 自定义session扫描器的实现 * 手动的 自己去管理 session 对象, 如果某个session,5分钟没有被访问过,那么就销毁 * * * 1、监听session对象的修改时间,要设置监听器:HttpSessionListener(session产生和销毁时) * 注意:定义一个容器,用来装session对象,在session产生时,添加到容器中; * 在session销毁时,从容器中移除;由此进行管理 * 2、设置一个定时器(timer),每隔5分钟进行检查一次,看看哪些session是超过5分钟没有被访问过, * 如果没有,则销毁 * 3、定时器timer有个任务对象,我们需要自己去创建这个任务对象:遍历检查session中超过5分钟的情况 * 而这个任务对象(TimerTask)是Runnable接口的实现类,所以只需实现Runnable的接口即可 * 4、定时器的启动需要进行设置,因此还要设置一个监听器:ServletContextListener; * 目的:设置监听器,web应用启动时开始工作,然后每隔5分钟检查一次 * *小结: * 1. 增删频繁的时候, 使用 LinkedList 性能好 * 2. 如何将一个list 变为一个线程安全的list,使用Collections.synchronizedList * 3. 定时器的使用 --- Timer 类 * 4. 遍历list集合的时候, 同时还要从list中去移除 元素, 避免 并发修改的异常(用ListIterator,而不是Iterator) * 5. 如何实现两段不同的代码 互斥,锁的使用 */ /* * * (1)实现HttpSessionListener监听器,实现两个方法 * public void sessionCreated(HttpSessionEvent se) {} * public void sessionDestroyed(HttpSessionEvent se) {} * * (2)实现ServletContextListener监听器,实现两个方法 * public void contextInitialized(ServletContextEvent sce) {} * public void contextDestroyed(ServletContextEvent sce) {} */ //监听器(观察者) @WebListener public class SessionScanner implements HttpSessionListener, ServletContextListener {// 自定义session扫描器的实现 // 定义一个容器, 将 每次 创建的session 对象放到 容器中去 // private List<HttpSession> list=new LinkedList<HttpSession>();//线程不安全 private List<HttpSession> list = Collections .synchronizedList(new LinkedList<HttpSession>());// 线程安全 public Object lock = new Object();// 定义一个锁,用于解决线程安全问题 // 主要解决:本对象的sessionCreated方法添加session,而MyTimerTask对象中方法销毁对象时,使用的是同一个session容器, // 这样,对同一个容器做不同的操作,肯能产生线程安全问题,所以要定义锁:lock,去解决这个线程安全问题 @Override // 事件对象(封装 session事件源 ) public void sessionCreated(HttpSessionEvent se) { System.out.println("执行了,说明新创建了一个session对象。。。"); HttpSession session = se.getSession();// 获取事件对象 // 定义锁,解决线程安全问题 synchronized (lock) { list.add(session);// 穿件的session放到容器中去 } // long lastAccessedTime = session.getLastAccessedTime();//最后一次修改时间 } @Override public void sessionDestroyed(HttpSessionEvent se) { System.out.println("执行了,说明销毁了一个session对象。。。"); } @Override public void contextInitialized(ServletContextEvent sce) { System.out.println("contextInitialized.............."); // 定义一个定时器,并且在web应用启动时开始工作 Timer timer = new Timer(); // 安排指定的任务从指定的延迟后开始进行重复的固定延迟执行 // task:安排的任务 // delay:举例开始的指定的延时时间 // period:重复时间 //立刻 启动 定时器, 每隔 5 分钟 重复 执行 timer.schedule(new MyTimerTask(list,lock), 0, 1000 * 60 * 5);// 1000毫秒*60*5=5分钟 } @Override public void contextDestroyed(ServletContextEvent sce) { System.out.println("contextDestroyed......"); } }
声明:该文观点仅代表作者本人,牛骨文系教育信息发布平台,牛骨文仅提供信息存储空间服务。
- 上一篇: 进程占用百分百CPU不卡
- 下一篇: C++实现用户自定义类
copyright © 2008-2019 亿联网络 版权所有 备案号:粤ICP备14031511号-2