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/