Session扫描器

这里定义了一个session扫描器,可以用于监控在线用户数量及踢出长时间没有访问操作页面的用户,销毁器session.

这里主要用到的技术有:

定时器:在监控到context创建时,启动一个定时器,每隔一段时间扫描一次所有在线用户。

监听器:实现一个监听器接口,此此接口实现了HttpSessionListener和ServletContextListener两个监听器接口。分别用于监听session的创建和退出(用户登录,用户退出)及web应用的启动和关闭。

多线程同步问题:多线程并发访问,容易造成共享数据的访问和操作冲突。

这里实现的session扫描器,使用的是将所有session保存到一个list中,但是对list的增加删除操作在多线程中,会存在并发访问冲突问题,java提供了Collections.synchronizedList方法可以使创建list的操作不会有并发访问冲突问题,它是一个多线程安全的list。

 List<HttpSession> lst = Collections.synchronizedList(new LinkedList<HttpSession>());

在创建一个定时器时,需要创建一个任务,此任务将会在定时器每隔指定的时间执行一次。这个任务也是一个类,此类实现了TimerTask接口run方法。需要执行的扫描操作都有run执行。

run中扫描list会和list的add方法存在多线程并发访问冲突问题,当list add一个session时,此时刚好定时器已经执行了run方法,此run方法中获取到的list并不知道list中有新session的加入,因此需要创建一个锁来保证此两部分数据的同步。另外还有一个办法,就是使扫描list和向list中添加一个session都是来源一个对象。  

下面是示例代码:

public class SessionScaner implements ServletContextListener,
		HttpSessionListener {
	
	private Timer timer;
	
	//通过Collections.synchronizedList创建的List能够解决向list中添加和删除某个元素
	//多线程并发问题
	private List<HttpSession> lst = Collections.synchronizedList(new LinkedList<HttpSession>());
	
	//通过此锁能够来实现代码块同步问题,信号量锁
	private Object lock = new Object();
	
	@Override
	public void sessionCreated(HttpSessionEvent se) {
		// TODO Auto-generated method stub
		HttpSession session = (HttpSession)se.getSource();
		synchronized (lock) {
			lst.add(session);
		}
		System.out.println("session被创建");
	}

	@Override
	public void sessionDestroyed(HttpSessionEvent se) {
		System.out.println("session被销毁了了!!");
	}

	@Override
	public void contextInitialized(ServletContextEvent sce) {
		// TODO Auto-generated method stub
		//web应用启动时,创建一个定时器,每个15秒扫描一次
		timer = new Timer();
		timer.schedule(new MyTask(lst, lock), 0 , 1000*5);
		
	}

	@Override
	public void contextDestroyed(ServletContextEvent sce) {
		// TODO Auto-generated method stub
		if(timer!=null){
			timer.cancel();
		}
	}

}

class MyTask extends TimerTask{
	private Object lock;
	private List lst;
	
	@Override
	public void run() {
		System.out.println("定时器执行了");
		// TODO Auto-generated method stub
		//通过lock锁进行同步,扫描lst中所有的session,15秒钟还没访问的,需要将其踢出。
		synchronized (lock) {
			ListIterator<HttpSession> it = lst.listIterator();
			while(it.hasNext()){
				HttpSession session = (HttpSession)it.next();
				if(System.currentTimeMillis()-session.getLastAccessedTime()>1000*5){
					session.invalidate();
					it.remove();	
				}
			}
		}	
	}
	
	MyTask(List lst, Object lock){
		this.lst = lst;
		this.lock = lock;
	}
	
}

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值