Comet push技术最根本的还是socket通信。它将客户端会话注册到一个map中,服务端通过客户端传入的JESSIONID来识别客户端,服务端不断给map中所有JESSIONID下的session注入数据。另一方面一旦有客户端链接,服务端就可以通过JESSIONID不断读session数据,然后发送给客户端。当然这种技术有缺陷,不适合高并发的访问。因为客户端和服务器端建立的是长连接,一旦连接过多服务端很难承受。
这里分服务端和客户端分别说明:
服务端
初始化阶段:
启动tomcat服务时,利用监听启动serversocket接口,并监听客户端请求 。同时还需启动一个session监听接口,当客户端访问时,session监听器通过sessionMap,将客户端发送关来的会话session中JSESSIONID作为sessionMap的key,session作为value.其中JSESSIONID是客户端的唯一标识。Sesssion监听程序大概如下:
public class CometSessionListener implements javax.servlet.http.HttpSessionListener{
public void sessionCreated(HttpSessionEvent event) {
SessionQueue.put(event.getSession().getId(), event.getSession());
}
public void sessionDestroyed(HttpSessionEvent event) {
System.out.println("sessionid="+event.getSession().getId());
SessionQueue.remove(event.getSession().getId());
}
}
数据推送到客户端阶段:
下面两个步骤是可以同时进行的
1.数据存放到sessionMap
sessionMap是保存发往每个客户端数据的容器。一旦服务端有新数据data,则将新数据data写入到每个客户端的session(即sessionMap.get(JSESSIONID))队列中。代码表示大概如下:
//获得sessionMap
LinkedHashMap sessions = SessionQueue.getSession(); Collection c = sessions.values();
Iterator iter = c.iterator();
while (iter.hasNext()) {
//通过循环获得每个session的值
HttpSession session = (HttpSession) iter.next();
PriorityQueue<Message> queue = (PriorityQueue<Message>) session
.getAttribute("message");
if (queue == null) {
queue = new PriorityQueue<Message>();
//将队列PriorityQueue赋值给session,session名为message
session.setAttribute("message", queue);
}
//将消息写入队列
queue.add(msg);
}
2.数据从sessionMap中取数据
当服务端socket监听到有客户端连接时,启动一个线程来处理,在此线程的run方法中,首先获取客户端http头部信息,解析出JESSIONID的值,然后通过死循环来不断取sessionMap中JESSSIONID下的消息,然后发送给客户端,当然发送给客户端之前的数据最终要转化为js,通过调用客户端js来改变html页面。代码表示大概如下:
HttpSession session = SessionQueue.get(getSid());
queue = (PriorityQueue<Message>) session
.getAttribute("message");
Message msg = null;
try {
msg = queue.remove();
} catch (Exception e) {
msg = null;
}
if (msg != null) {
OutputStream out = sc.getOutputStream();
out.write("HTTP/1.1 200 OK\r\n".getBytes());
out.write("Content-Type:text/html;charset=GBK\r\n\r\n".getBytes());
String content = "<script language='javascript'>window.parent.showMsg({user:'"
+ msg.getUser()
+ "',msg:'"
+ msg.getMsg() + "'});</script>";
out.write(content.getBytes());
out.flush();
}
1. 客户端
客户端就比较简单了,因为它是被动的,所以只要在页面初始化时发送一个http请求到服务端。建立socket连接就行了。然后就是页面上要有个js来改变页面的数据。代码大概如下:
<script>
var sId = getCookie("JSESSIONID");
window.onload = function(){
listener.action = listener.action + "&JSESSIONID="+sId;
listener.submit();
}
function showMsg(msg){
var msgDiv = document.createElement("div");
msgDiv.innerHTML = msg.user + " say: " + msg.msg;
document.all.info.appendChild(msgDiv);
}
function getCookie(sName){
var aCookie = document.cookie.split("; ");
for (var i=0; i < aCookie.length; i++){
var aCrumb = aCookie[i].split("=");
if (sName == aCrumb[0])
return unescape(aCrumb[1]);
}
return null;
}
</script>
<iframe name="result" width=0 height=0>
</iframe>
<iframe name="send">
</iframe>
<form method="post" name="listener" target="result"
action="http://localhost:81/index.jsp?callback=window.parent.showMsg">
</form>
注:在这学习的过程中是参照牛人hexiaodong的代码,其blog地址http://hexiaodong.iteye.com/