在一些需要实时消息功能的网站应用中,除了客户端轮询请求服务器获取消息外,还有一种方案就是comet长连接推送消息。显然后者更具有优势,实时性高,客户端、服务端压力都比较小。对于长连接方案,我们需要考虑对长连接进行管理,以便有消息时可以推送到客户端。可是对于大量长连接的建立服务器是有可能会被压垮的,所以不是一味的接受连接和hold住连接就可以的,需要对连接进行超时管理,如果超时需要推送重连消息或是关闭连接。我们可以初略分析下具体的管理逻辑。

       先简单分析下具体的应用场景:用户登录一个电商网站(苏宁,国美),恰好在大量客户登录之后,商家(海尔、格力等)推出了一系列优惠活动(针对注册用户),需要及时推送到浏览器让用户了解,不需要用户刷新页面。

      就这么一个常见的功能,我们需要怎么做呢?首先,用户打开页面,客户端js就需要请求长连接,并携带用户信息(id)过来,服务器使用list、queue之类的存储起来,记录当时的请求时间;另起一个守护线程,扫描是否有超时或需要断开的连接,超时的需要发重连消息,断开的发断开消息;如果有消息需要推送从list、queue中获取连接推送即可。好像没什么问题,对于超时处理,遍历大量元素耗时太长是接受不了的,那么排除list存储,queue具有FIFO的特性,每次检查超时,只检查表头元素即可,这样是很快。但是忽略了重连这个问题,间隔时间内重连,不能重新new一个连接对象(大量新对象耗内存伤不起),只能复用之前的对象,之前对象的time就会发生变化,queue就会变成不按时间排序的有序队列了,如果按之前的处理方式就会阻塞在表头,无法处理了,并且queue的重连遍历也是个问题,队列本身遍历比较麻烦。从现在来看单一的数据结构很难处理这样的问题,需要多个数据结构结合使用。我们知道在linux内核中定时器的实现采用wheel。我们可以借鉴之,对于超时的处理我们使用hashedwheel,hashed+wheel,hash主要用于对时间进行处理,决定放在wheel的那个区间段中;wheel是一个数组,减小了遍历的数目。

        关于HashedWheelTimer实现具体见代码。netty中有类似的实现,这里进行了简化。主要有:1. 守护线程由外部提供,该线程来调用hashedwheel;2. 假定我们的超时时间不会大于wheel的圈时长(假设wheel 1s转动一次,一共512个区间,圈时长就为512s,我们的超时时间不会这么久),就不会有第几圈计算的问题;


package com.yq.algorithmic;

/**
 * 
 * 
 * 描述:连接管理
 * 创建时间:2011-9-16下午08:42:34
 * @author yq76034150
 *
 */
public class TimeOut {
	final long time;
	final long deadline;
	private int rounds;
	
	//连接的用户id
	//连接的channel
	
	public TimeOut(long time, long deadline) {
		super();
		this.time = time;
		this.deadline = deadline;
	}

	public int getRounds() {
		return rounds;
	}

	public void setRounds(int rounds) {
		this.rounds = rounds;
	}

	public long getTime() {
		return time;
	}

	public long getDeadline() {
		return deadline;
	}
	
	public String toString(){
		return "survival: " + (System.currentTimeMillis() - time);
	}
}



package com.yq.algorithmic;

import java.util.Queue;
import java.util.concurrent.ConcurrentLinkedQueue;

/**
 * 
 * 描述:一个基于时间hash的wheel超时器
 * 创建时间:2011-9-16上午11:44:02
 * @author yq76034150
 *
 */
public class HashedWheelTimer {
	volatile int tick = 1;
	volatile int currentWheel;
	private Queue<TimeOut>[] wheel;
	final int mask;
	private long delay;
	//private int ticksPerWheel;
	private long tickDuration;
	//private long roundDuration;
	
	/**
	 * 描述:
	 * @param ticksPerWheel 一圈多少tick
	 */
	public HashedWheelTimer(int ticksPerWheel, long tickDuration, long delay){
		mask = ticksPerWheel - 1;
		this.delay = delay; 
		//this.ticksPerWheel = ticksPerWheel;
		this.tickDuration = tickDuration;
		//this.roundDuration = tickDuration * ticksPerWheel;
		createWheel(ticksPerWheel);
	}
	
	/**
	 * 
	 * 描述:wheel中填充数据
	 * @param timeout
	 */
	public void newTimeOut(TimeOut timeout){
		long shift = delay / tickDuration;
		//long remainRounds = delay / roundDuration;
		int stopIndex = currentWheel + (int)shift & mask;
		
		wheel[stopIndex].offer(timeout);
	}
	
	/**
	 * 
	 * 描述:外部线程调用,外部线程调用间隔必须和tickDuration一致
	 */
	public void run(long start){
		//模拟外部执行线程执行间隔数
		try {
			Thread.sleep(tickDuration);
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		tick++;
		currentWheel = currentWheel + 1 & mask;
		long deadline = start + tick * tickDuration;
		
		Queue<TimeOut> queue = wheel[currentWheel];
		TimeOut timeOut = queue.peek();
		if(timeOut != null){
			long firstDeadline = timeOut.getDeadline();
			boolean isDead = firstDeadline <= deadline;
			if(firstDeadline <= deadline){
				while(isDead){ //对一个tick中的队列元素递归检查超时
					queue.remove();
					System.out.println("delete: " + currentWheel + timeOut);
					
					timeOut = queue.peek();
					if(timeOut != null){
						firstDeadline = timeOut.getDeadline();
						isDead = firstDeadline <= deadline;
					} else {
						break;
					}
				}
			}
			
		}
	}

	/**
	 * 
	 * 描述:初始化wheel
	 * @param ticksPerWheel
	 */
	private void createWheel(int ticksPerWheel) {
		wheel = new Queue[ticksPerWheel];
		for(int i = 0; i < ticksPerWheel; i++){
			wheel[i] = new ConcurrentLinkedQueue<TimeOut>();
		}
	}

	/**
	 * 描述:
	 * @param args
	 */
	public static void main(String[] args) {
		long tickDuration = 1000;
		long delay = 9000;
		HashedWheelTimer timer = new HashedWheelTimer(4, tickDuration, delay);
		
		long start = System.currentTimeMillis();
		TimeOut timeout = new TimeOut(start, start + delay);
		timer.newTimeOut(timeout);

		timer.run(start);
				
		start = System.currentTimeMillis();
		timeout = new TimeOut(start, start + delay);
		timer.newTimeOut(timeout);
		
		timer.run(start);
		
		for(int i = 0; i < 10; i++){
			timer.run(start);
		}
	}

}



上面的例子使用queue,会存在阻塞问题。测试代码如下:

package com.yq.hash.timer;

import java.util.Queue;
import java.util.concurrent.ConcurrentLinkedQueue;

/**
 * 
 * 描述:一个基于时间hash的wheel超时器
 * 创建时间:2011-9-16上午11:44:02
 * @author yq76034150
 *
 */
public class HashedWheelTimer {
	volatile int tick = 1;
	volatile int currentWheel;
	private Queue<TimeOut>[] wheel;
	final int mask;
	private long delay;
	//private int ticksPerWheel;
	private long tickDuration;
	//private long roundDuration;
	
	/**
	 * 描述:
	 * @param ticksPerWheel 一圈多少tick
	 */
	public HashedWheelTimer(int ticksPerWheel, long tickDuration, long delay){
		mask = ticksPerWheel - 1;
		this.delay = delay; 
		//this.ticksPerWheel = ticksPerWheel;
		this.tickDuration = tickDuration;
		//this.roundDuration = tickDuration * ticksPerWheel;
		createWheel(ticksPerWheel);
	}
	
	/**
	 * 
	 * 描述:wheel中填充数据
	 * @param timeout
	 */
	public void newTimeOut(TimeOut timeout){
		long shift = delay / tickDuration;
		//long remainRounds = delay / roundDuration;
		int stopIndex = currentWheel + (int)shift & mask;
		
		wheel[stopIndex].offer(timeout);
	}
	
	/**
	 * 
	 * 描述:外部线程调用,外部线程调用间隔必须和tickDuration一致
	 */
	public void run(long start){
		//模拟外部执行线程执行间隔数
		try {
			Thread.sleep(tickDuration);
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		tick++;
		currentWheel = currentWheel + 1 & mask;
		long deadline = start + tick * tickDuration;
		
		Queue<TimeOut> queue = wheel[currentWheel];
		TimeOut timeOut = queue.peek();
		if(timeOut != null){
			long firstDeadline = timeOut.getDeadline();
			boolean isDead = firstDeadline <= deadline;
			//if(firstDeadline <= deadline){
				while(isDead){ //对一个tick中的队列元素递归检查超时
					queue.remove();
					System.out.println("tick: " + tick + " delete: " + currentWheel + timeOut);
					
					timeOut = queue.peek();
					if(timeOut != null){
						firstDeadline = timeOut.getDeadline();
						isDead = firstDeadline <= deadline;
					} else {
						break;
					}
				}
			//}
			
		}
	}

	/**
	 * 
	 * 描述:初始化wheel
	 * @param ticksPerWheel
	 */
	private void createWheel(int ticksPerWheel) {
		wheel = new Queue[ticksPerWheel];
		for(int i = 0; i < ticksPerWheel; i++){
			wheel[i] = new ConcurrentLinkedQueue<TimeOut>();
		}
	}

	/**
	 * 描述:
	 * @param args
	 */
	public static void main(String[] args) {
		long tickDuration = 1000;
		long delay = 9000;
		HashedWheelTimer timer = new HashedWheelTimer(4, tickDuration, delay);
		
		long start = System.currentTimeMillis();
		TimeOut timeout = new TimeOut(start, start + delay, 1);
		timer.newTimeOut(timeout);
		timer.run(start);
		
		//4s后 保证和之前的落在一个区间
		for(int i = 0; i < 3; i++){
			timer.run(start);
		}

		TimeOut timeout2 = new TimeOut(System.currentTimeMillis(), System.currentTimeMillis() + delay, 2);
		timer.newTimeOut(timeout2);
		timer.run(start);
			
		//更改表头的时间 queue结构的就会阻塞
		timeout.setTime(start + 19000);
		timeout.setDeadline(start + 19000 + delay);
		timer.newTimeOut(timeout);
		timer.run(start);
		
		for(int i = 0; i < 30; i++){
			timer.run(start);
		}
	}

}



会发现 ,超过9s后才被删除。


tick: 30 delete: 1v:1316356889764 time:1316356879760 deadline: 1316356888760 survival: 10004
tick: 30 delete: 1v:1316356889765 time:1316356864763 deadline: 1316356873763 survival: 25002
tick: 31 delete: 2v:1316356890765 time:1316356879760 deadline: 1316356888760 survival: 11005



将queue改为map后正常。


package com.yq.hash.timer;

import java.util.Map;
import java.util.Map.Entry;
import java.util.Queue;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentLinkedQueue;

/**
 * 
 * 描述:一个基于时间hash的wheel超时器
 * 创建时间:2011-9-16上午11:44:02
 * @author yq76034150
 *
 */
public class HashedWheelTimer {
	volatile int tick = 1;
	volatile int currentWheel;
	private Map<TimeOut, Boolean>[] wheel;
	final int mask;
	private long delay;
	//private int ticksPerWheel;
	private long tickDuration;
	//private long roundDuration;
	
	/**
	 * 描述:
	 * @param ticksPerWheel 一圈多少tick
	 */
	public HashedWheelTimer(int ticksPerWheel, long tickDuration, long delay){
		mask = ticksPerWheel - 1;
		this.delay = delay; 
		//this.ticksPerWheel = ticksPerWheel;
		this.tickDuration = tickDuration;
		//this.roundDuration = tickDuration * ticksPerWheel;
		createWheel(ticksPerWheel);
	}
	
	/**
	 * 
	 * 描述:wheel中填充数据
	 * @param timeout
	 */
	public void newTimeOut(TimeOut timeout){
		long shift = delay / tickDuration;
		//long remainRounds = delay / roundDuration;
		int stopIndex = currentWheel + (int)shift & mask;
		
		wheel[stopIndex].put(timeout, true);
	}
	
	/**
	 * 
	 * 描述:外部线程调用,外部线程调用间隔必须和tickDuration一致
	 */
	public void run(long start){
		//模拟外部执行线程执行间隔数
		try {
			Thread.sleep(tickDuration);
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		tick++;
		currentWheel = currentWheel + 1 & mask;
		long deadline = start + tick * tickDuration;
		
		Map<TimeOut, Boolean> map = wheel[currentWheel];
		
		TimeOut timeOut = null;
		for(Entry<TimeOut, Boolean> entry : map.entrySet()){
			timeOut = entry.getKey();
			if(timeOut.getDeadline() <= deadline){
				map.remove(timeOut);
				System.out.println("tick: " + tick + " delete: " + currentWheel + timeOut);
			}
		}

	}

	/**
	 * 
	 * 描述:初始化wheel
	 * @param ticksPerWheel
	 */
	private void createWheel(int ticksPerWheel) {
		wheel = new Map[ticksPerWheel];
		for(int i = 0; i < ticksPerWheel; i++){
			wheel[i] = new ConcurrentHashMap<TimeOut, Boolean>();
		}
	}

	/**
	 * 描述:
	 * @param args
	 */
	public static void main(String[] args) {
		long tickDuration = 1000;
		long delay = 9000;
		HashedWheelTimer timer = new HashedWheelTimer(4, tickDuration, delay);
		
		long start = System.currentTimeMillis();
		TimeOut timeout = new TimeOut(start, start + delay, 1);
		timer.newTimeOut(timeout);
		timer.run(start);
		
		//4s后 保证和之前的落在一个区间
		for(int i = 0; i < 3; i++){
			timer.run(start);
		}

		long start2 = System.currentTimeMillis();
		TimeOut timeout2 = new TimeOut(start2, start2 + delay, 2);
		timer.newTimeOut(timeout2);
		timer.run(start);
			
		//更改表头的时间 queue结构的就会阻塞
		timeout.setTime(start + 30000);
		timeout.setDeadline(start + 30000 + delay);
		timer.newTimeOut(timeout);
		timer.run(start);
		
		for(int i = 0; i < 50; i++){
			timer.run(start);
		}
	}

}



在同一区间段内,先删除2 再删除1,而不是等删除1后再删除2.