Ajax轮询以及Comet模式—写在Servlet 3.0发布之前

2024-04-17 00:32

本文主要是介绍Ajax轮询以及Comet模式—写在Servlet 3.0发布之前,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

来源:http://www.blogjava.net/rosen/archive/2009/02/11/254309.html

2008 年的夏天,偶然在网上闲逛的时候发现了 Comet 技术,人云亦云间,姑且认为它是由 Dojo Alex Russell 2006 年提出。在阅读了大量的资料后,萌发出写篇 blog 来说明什么是 Comet 的想法。哪知道这个想法到了半年后的今天才提笔,除了繁忙的工作拖延外,还有 Comet 本身带来的困惑。

Comet 能带来生产力的提升是有目共睹的。现在假设有 1000 个用户在使用某软件,轮询 (polling) Comet 的设定都是 1s 10s 100s 的潜伏期,那么在相同的潜伏期内, Comet 所需要的带宽更小,如下图:

 

 

不仅仅是在带宽上的优势,每个用户所真正感受到的响应时间(潜伏期)更短,给人的感觉也就更加的实时,如下图:

 

 

再引用一篇 IBMDW 上的译文《使用 Jetty Direct Web Remoting 编写可扩展的 Comet 应用程序》,其中说到:吸引人们使用 Comet 策略的其中一个优点是其显而易见的高效性。客户机不会像使用轮询方法那样生成烦人的通信量,并且事件发生后可立即发布给客户机。

上面一遍一遍的说到 Comet 技术的优势,那么我们可以替换现有的技术结构了?不幸的是,近半年的擦边球式的关注使我对 Comet 的理解越发的糊涂,甚至有人说 Comet 这个名词已被滥用。去年的一篇博文,《 The definition of Comet? 》使 Comet 更加扑朔迷离,甚至在维基百科上大家也对准确的 Comet 定义产生争论。还是等牛人们争论清楚再修改维基百科吧,在这里我想还是引用维基百科对 Comet 的定义:服务器推模式 (HTTP server push streaming) 以及长轮询 (long polling) ,这两种模式都是 Comet 的实现。

除了对 Comet 的准确定义尚缺乏有效的定论外, Comet 还存在不少技术难题,随着 Tomcat 6 Jetty 6 的发布,他们基于 NIO 各自实现了异步 Servlet 机制。有兴趣的看官可以分别实现这两个容器的 Comet ,至少我还没玩转。

在编写服务器端的代码上面,我很困惑, http://tomcat.apache.org/tomcat-6.0-doc/aio.html 这里演示了如何在 Tomcat 6 中实现异步 Servlet ;我们再把目光换到 Jetty 6 上,还是前面提到的那篇 IBMDW 译文,如果你和我一样无聊,可以下载那边文章的 sample 代码。我惊奇的发现每个厂商对异步 Servlet 的封装是不同的,一个傻傻的问题:我的 Comet 服务器端的代码可移植么?至今我还在问这个问题!好吧,业界有规范么?有当然有,不过看起来有些争论会发生——那就是 Servlet 3.0 规范 (JSR-315) Servlet 3.0 正在公开预览,它明确的支持了异步 Servlet ,《 Servlet 3.0 公开预览版引发争论》,又让我高兴不起来了:“来自 RedHat Bill Burke 写的一篇博文,其中他批评了 Jetty 6 中的异步 servlet 实现 ......Greg Wilkins 宣布他致力于 Servlet 3.0 异步 servlet 的一个实现 ...... 虽然还需要更多测试,但是这个代码已经实现了基本的异步行为,不需要很复杂的重新分发请求或者前递方法。我相信这代表了 3.0 的合理折中方案。在我们从 3.0 的简单子集里获得经验之后,如果需要更多的特性,可以添加到 3.1 ........” 。牛人们还在做最佳范例,口水仗也还要继续打,看来要尝到 Comet 的甜头是很困难的。 STOP !我已经不想再分析如何写客户端的代码了,什么 dojo extJs DWR ZK....... 都有自己的实现。我认为这一切都要等 Servelt 3.0 正式发布以后,如何编写客户端代码才能明朗点。

现在抛开绕来绕去的争执吧,既然 Ajax+Servlet 实现 Comet 很困难,何不换个思维呢。我这里倒是有个小小的 sample ,说明如何在 Adobe BlazeDS 中实现长轮询模式。关于 BlazeDS ,可以在这里找到些信息。为了说明什么是长轮询,首先来看看什么是轮询,既在一定间隔期内由 web 客户端发起请求到服务器端取回数据,如下图所示:

                         

 

 

至于轮询的缺点,在前面的论述中已有覆盖,至于优点大家可以 google 一把,我觉得最大的优点就是技术上很好实现,下面是个 Ajax 轮询的例子,这是一个简单的聊天室,首先是 chat.html 代码,想必这些代码网上一抓就一大把,支持至少 IE6 IE7 FF3 浏览器,让人烦心的是乱码问题,在传递到 Servlet 之前要 encodeURI 一下


<! DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd" >
<!--
    chat page
    author rosen jiang
    since 2008/07/29
-->
< html >
  
< head >
   
< meta  http-equiv ="content-type"  content ="text/html; charset=utf-8" >
    
< script  type ="text/javascript" >
    
// servlets url
     var  url  =   " http://127.0.0.1:8080/ajaxTest/Ajax " ;
    
// bs version
     var  version  =  navigator.appName + "   " + navigator.appVersion;
    
// if is IE
     var  isIE  =   false ;

    
if (version.indexOf( " MSIE 6 " ) > 0   ||  version.indexOf( " MSIE 7 " ) > 0 ){
        isIE 
=   true ;
    }

    
// Httprequest object
     var  Httprequest  =   function () {}
    
// creatHttprequest function of Httprequest
    Httprequest.prototype.creatHttprequest = function (){
        
var  request  =   false ;
        
// init XMLHTTP or XMLHttpRequest
         if  (isIE) {
            
try  {
                request 
=   new  ActiveXObject( " Msxml2.XMLHTTP " );
            } 
catch  (e) {
                
try  {
                    request 
=   new  ActiveXObject( " Microsoft.XMLHTTP " );
                } 
catch  (e) {}
            }
        }
else  {  // Mozilla bs etc.
            request  =   new  XMLHttpRequest();
        }
        
if  ( ! request) {
            
return   false ;
        }
        
return  request;
    }
    
// sendMsg function of Httprequest
    Httprequest.prototype.sendMsg = function (msg){
        
var  http_request  =      this .creatHttprequest();
        
var  reslult  =   "" ;
        
var  methed  =   false ;
        
if  (http_request) {    
            
if  (isIE) {                
                http_request.onreadystatechange 
=
                        
function  (){ // callBack function
                             if  (http_request.readyState  ==   4 ) {
                                
if  (http_request.status  ==   200 ) {
                                    reslult 
=  http_request.responseText;
                                } 
else  {
                                    alert(
" 您所请求的页面有异常。 " );
                                }
                            }
                        };
            } 
else  {
                http_request.onload 
=  
                        
function  (){ //  callBack function of Mozilla bs etc.
                             if  (http_request.readyState  ==   4 ) {
                                
if  (http_request.status  ==   200 ) {
                                    reslult 
=  http_request.responseText;
                                } 
else  {
                                    alert(
" 您所请求的页面有异常。 " );
                                }
                            }
                        };
            }
            
// send msg
             if (msg != null   &&  msg != "" ){
                request_url 
=  url + " ? " + Math.random() + " &msg= " + msg;
                
// encodeing utf-8 Character
                request_url  =  encodeURI(request_url);
                http_request.open(
" GET " , request_url,  false );
            }
else {
                http_request.open(
" GET " , url + " ? " + Math.random(),  false );
            }
            http_request.setRequestHeader(
" Content-type " , " charset=utf-8; " );
            http_request.send(
null );
        }
        
return  reslult;    
    }
</ script >
</ head >
< body >
  
< div >
      
< input  type ="text"  id ="sendMsg" ></ input >
      
< input  type ="button"  value ="发送消息"  onclick ="send()" />
      
< br />< br />
      
< div  style ="width:470px;overflow:auto;height:413px;border-style:solid;border-width:1px;font-size:12pt;" >
             
< div  id ="msg_content" ></ div >
          
< div  id ="msg_end"  style ="height:0px; overflow:hidden" >   </ div >
      
</ div >
  
</ div >
</ body >
< script  type ="text/javascript" >
    
var  data_comp  =   "" ;
    
// send button click
     function  send(){
        
var  sendMsg  =  document.getElementById( " sendMsg " );
        
var  hq  =   new  Httprequest();
        hq.sendMsg(sendMsg.value);
        sendMsg.value
= "" ;
    }
    
// processing wnen message recevied
     function  writeData(){
        
var  msg_content  =  document.getElementById( " msg_content " );
        
var  msg_end  =  document.getElementById( " msg_end " );
        
var  hq  =   new  Httprequest();
        
var  value  =  hq.sendMsg();
        
if (data_comp  !=  value){
            data_comp 
=  value;
            msg_content.innerHTML 
=  value;
            msg_end.scrollIntoView();
        }
        setTimeout(
" writeData() " 1000 );
    }
    
// init load writeData 
    onload  =  writeData;
</ script >
</ html >

 

 

 

接下来是 Servlet ,如果你是用的 Tomcat ,在这里注意下编码问题,否则又是乱码,另外我使用 LinkedList 实现了一个队列,该队列的最大长度是 30 ,也就是最多能保存 30 条聊天信息,旧的将被丢弃,另外新的客户端进来后能读取到最近的信息:

package  org.rosenjiang.ajax;

import  java.io.IOException;
import  java.io.PrintWriter;
import  java.text.SimpleDateFormat;
import  java.util.Date;
import  java.util.LinkedList;

import  javax.servlet.ServletException;
import  javax.servlet.http.HttpServlet;
import  javax.servlet.http.HttpServletRequest;
import  javax.servlet.http.HttpServletResponse;

/**
 * 
 * 
@author  rosen jiang
 * 
@since  2009/02/06
 * 
 
*/
public   class  Ajax  extends  HttpServlet {
    
private   static   final   long  serialVersionUID  =   1L ;
    
//  the length of queue
     private   static   final   int  QUEUE_LENGTH  =   30 ;
    
//  queue body
     private   static  LinkedList < String >  queue  =   new  LinkedList < String > ();
    
    
/**
     * response chat content
     * 
     * 
@param  request
     * 
@param  response
     * 
@throws  ServletException
     * 
@throws  IOException
     
*/
    
public   void  doGet(HttpServletRequest request, HttpServletResponse response)
            
throws  ServletException, IOException {
        
// parse msg content
        String msg  =  request.getParameter( " msg " );
        SimpleDateFormat sdf 
=   new  SimpleDateFormat( " yyyy-MM-dd HH:mm:ss " );
        
// push to the queue
         if  (msg  !=   null   &&   ! msg.equals( "" )) {
            
byte [] b  =  msg.getBytes( " ISO_8859_1 " );
            msg 
=  sdf.format( new  Date())  + "    " + new  String(b,  " utf-8 " ) + " <br> " ;
            
if (queue.size()  ==  QUEUE_LENGTH){
                queue.removeFirst();
            }
            queue.addLast(msg);
        }
        
// response client
        response.setContentType( " text/html " );
        response.setCharacterEncoding(
" utf-8 " );
        PrintWriter out 
=  response.getWriter();
        msg 
=   "" ;
        
// loop queue
         for ( int  i = 0 ; i < queue.size(); i ++ ){
            msg 
=  queue.get(i);
            out.println(msg
== null   ?   ""  : msg);
        }
        out.flush();
        out.close();
    }

    
/**
     * The doPost method of the servlet.
     *
     * 
@param  request
     * 
@param  response
     * 
@throws  ServletException
     * 
@throws  IOException
     
*/
    
public   void  doPost(HttpServletRequest request, HttpServletResponse response)
            
throws  ServletException, IOException {
        
this .doGet(request, response);
    }
}

 

 

 

 

打开浏览器,实验下效果,将就用吧,稍微有些延迟。还是看看长轮询吧,长轮询有三个显著的特征:

1. 服务器端会阻塞请求直到有数据传递或超时才返回。

2. 客户端响应处理函数会在处理完服务器返回的信息后,再次发出请求,重新建立连接。

3. 当客户端处理接收的数据、重新建立连接时,服务器端可能有新的数据到达;这些信息会被服务器端保存直到客户端重新建立连接,客户端会一次把当前服务器端所有的信息取回。

 

 

下图很好的说明了以上特征:

                             

 

 

既然关注的是 BlazeDS 如何实现长轮询,那么有必要稍微了解下。 BlazeDS 包含了两个重要的服务,进行远端方法调用的 RPC service 和传递异步消息的 Messaging Service ,我们即将探讨的长轮询属于 Messaging Service Messaging Service 使用 producer consumer 模式来分别定义消息的发送者 (producer) 和消费者 (consumer) ,具体到 Flex 代码,有 Producer Consumer 两个组件对应。在广阔的互联网上有很多 BlazeDS 入门的中文教材,我就不再废话了。假设你已经装好 BlazeDS ,打开 WEB-INF/flex/services-config.xml 文件,在 channels 节点内加一个 channel 声明长轮询频道,关于 channel endpoint 请参阅 About channels and endpoints 章节:


         < channel-definition  id ="long-polling-amf"  class ="mx.messaging.channels.AMFChannel" >
            
< endpoint  url ="http://{server.name}:{server.port}/{context.root}/messagebroker/longamfpolling"  class ="flex.messaging.endpoints.AMFEndpoint" />
            
< properties >
                
< polling-enabled > true </ polling-enabled >
                
< wait-interval-millis > 60000 </ wait-interval-millis >
                
< polling-interval-millis > 0 </ polling-interval-millis >
                
< max-waiting-poll-requests > 150 </ max-waiting-poll-requests >
            
</ properties >
    
</ channel-definition >

 

如何实现长轮询的玄机就在上面的 properties 节点内, polling-enabled = true ,打开轮询模式; wait-interval-millis = 6000 服务器端的潜伏期,也就是服务器会保持与客户端的连接,直到超时或有新消息返回(恩,看来这就是长轮询了); polling-interval-millis = 0 表示客户端请求服务器端的间隔期, 0 表示没有任何的延迟; max-waiting-poll-requests = 150 表示服务器能承受的最大长连接用户数,超过这个限制,新的客户端就会转变为普通的轮询方式(至于这个数值最大能有多大,这和你的 web 服务器设置有关了,而 web 服务器的最大连接数就和操作系统有关了,这方面的话题不在本文内探讨)。

 

 

其实这样设置之后,长轮询的代码已经实现了一半了。恩,不错!看起来比异步 Servlet 实现起来简单多了。不过要实现和之前 Ajax 轮询一样的效果,还得实现自己的 ServiceAdapter ,这就是 Adapter 的用处:


package  org.rosenjiang.flex;

import  java.text.SimpleDateFormat;
import  java.util.Date;
import  java.util.LinkedList;

import  flex.messaging.io.amf.ASObject;
import  flex.messaging.messages.Message;
import  flex.messaging.services.MessageService;
import  flex.messaging.services.ServiceAdapter;

/**
 * 
 * 
@author  rosen jiang
 * 
@since  2009/02/06
 * 
 
*/
public   class  MyMessageAdapter  extends  ServiceAdapter {

    
//  the length of queue
     private   static   final   int  QUEUE_LENGTH  =   30 ;
    
//  queue body
     private   static  LinkedList < String >  queue  =   new  LinkedList < String > ();

    
/**
     * invoke method
     * 
     * 
@param  message Message
     * 
@return  Object
     
*/
    
public  Object invoke(Message message) {
        SimpleDateFormat sdf 
=   new  SimpleDateFormat( " yyyy-MM-dd HH:mm:ss " );
        MessageService msgService 
=  (MessageService) getDestination()
            .getService();
        
// message Object
        ASObject ao  =  (ASObject) message.getBody();
        
// chat message
        String msg  =  (String) ao.get( " chatMessage " );
        
if  (msg  !=   null   &&   ! msg.equals( "" )) {
            msg 
=  sdf.format( new  Date())  +   "    "   +  msg  +   " /r " ;
            
if (queue.size()  ==  QUEUE_LENGTH){
                queue.removeFirst();
            }
            queue.addLast(msg);
        }
        msg 
=   "" ;
        
// loop queue
         for ( int  i = 0 ; i < queue.size(); i ++ ){
            String chatData 
=  queue.get(i);
            
if  (chatData  !=   null ) {
                msg 
+=  chatData;
            }
        }
        ao.put(
" chatMessage " , msg);
        message.setBody(ao);
        msgService.pushMessageToClients(message, 
false );
        
return   null ;
    }
}

 

接下来注册该 Adapter ,打开 WEB-INF/flex/messaging-config.xml 文件,在 adapters 节点内加入一个 adapter-definition 来声明自定义 Adapter


< adapter-definition  id ="myad"  class ="org.rosenjiang.flex.MyMessageAdapter" />

 

 

 

接着定义一个 destination ,以便 Flex 客户端能订阅聊天室,组装好之前定义的长轮询频道和 adapter

 


     < destination  id ="chat" >
        
< channels >
            
< channel  ref ="long-polling-amf" />
        
</ channels >
        
< adapter  ref ="myad" />
    
</ destination >

 

服务器端就算搞定了,接着搞定 Flex 那边的代码吧,灰常灰常的简单。先到 Building your client-side application 学习如何创建和 BlazeDS 通讯的 Flex 项目。然后在 chat.mxml 中写下:


<? xml version="1.0" encoding="utf-8" ?>
< mx:Application  xmlns:mx ="http://www.adobe.com/2006/mxml"  creationComplete ="consumer.subscribe();send()" >
    
    
< mx:Script >
        
<![CDATA[
        
            import mx.messaging.messages.AsyncMessage;
            import mx.messaging.messages.IMessage;
            
            private function send():void
            {
                var message:IMessage = new AsyncMessage();
                message.body.chatMessage = msg.text;
                producer.send(message);
                msg.text = "";
            }
                        
            private function messageHandler(message:IMessage):void
            {
                log.text = message.body.chatMessage + "/n";
            }
            
        
]]>
    
</ mx:Script >
    
    
< mx:Producer  id ="producer"  destination ="chat" />
    
< mx:Consumer  id ="consumer"  destination ="chat"  message ="messageHandler(event.message)" />
    
    
< mx:Panel  title ="Chat"  width ="100%"  height ="100%" >
        
< mx:TextArea  id ="log"  width ="100%"  height ="100%" />
        
< mx:ControlBar >
             
< mx:TextInput  id ="msg"  width ="100%"  enter ="send()" />
             
< mx:Button  label ="Send"  click ="send()" />  
        
</ mx:ControlBar >
    
</ mx:Panel >
    
</ mx:Application >

 

 

 

之前我们说到的 Producer Consumer 组件在这里出现了,由于我们要订阅的是同一个聊天室,所以 destination="chat" ,而 Consumer 组件则注册回调函数 messageHandler() ,处理异步消息的到来。当打开这个聊天客户端的时候,在 creationComplete 初始化完成后,立即进行 consumer.subscribe() ,其实接下来应该就能直接收到服务器端回馈的聊天记录了,但是我没仔细学习如何监听客户端的订阅,所以在这里我直接 send() 了一个空消息以便服务器端能回馈已有的聊天记录,接下来我就不用再讲解了,都能看懂。

现在打开浏览器,感受下长轮询的效果吧。不过遇到个问题,如果 FF 同时开两个聊天窗口,第二个打开的会有延迟感, IE 也是,按照牛人们的说法,当一个浏览器开两个以上长连接的时候才会有延迟感,不解。 BlazeDS 的长轮询也不是十全十美,有人说它不是真正的“实时” The Truth About BlazeDS and Push Messaging ,随即引发出口水仗,里面提到的 RTMP 协议在 2009 1 月已开源,相信以后 BlazeDS 会更“实时”;接着又有人说 BlazeDS 不是非阻塞式的,这个问题后来也没人来对应。罢了,毕竟BlazeDS才开源不久,容忍一下吧。最后,我想说的是,不论 BlazeDS 到底有什么问题,至少实现起来是轻松的,在 Servlet 3.0 没发布之前,是个不错的选择。

 

请注意!引用、转贴本文应注明原作者:Rosen Jiang 以及出处: http://www.blogjava.net/rosen

这篇关于Ajax轮询以及Comet模式—写在Servlet 3.0发布之前的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/910330

相关文章

Servlet中配置和使用过滤器的步骤记录

《Servlet中配置和使用过滤器的步骤记录》:本文主要介绍在Servlet中配置和使用过滤器的方法,包括创建过滤器类、配置过滤器以及在Web应用中使用过滤器等步骤,文中通过代码介绍的非常详细,需... 目录创建过滤器类配置过滤器使用过滤器总结在Servlet中配置和使用过滤器主要包括创建过滤器类、配置过滤

高效+灵活,万博智云全球发布AWS无代理跨云容灾方案!

摘要 近日,万博智云推出了基于AWS的无代理跨云容灾解决方案,并与拉丁美洲,中东,亚洲的合作伙伴面向全球开展了联合发布。这一方案以AWS应用环境为基础,将HyperBDR平台的高效、灵活和成本效益优势与无代理功能相结合,为全球企业带来实现了更便捷、经济的数据保护。 一、全球联合发布 9月2日,万博智云CEO Michael Wong在线上平台发布AWS无代理跨云容灾解决方案的阐述视频,介绍了

在JS中的设计模式的单例模式、策略模式、代理模式、原型模式浅讲

1. 单例模式(Singleton Pattern) 确保一个类只有一个实例,并提供一个全局访问点。 示例代码: class Singleton {constructor() {if (Singleton.instance) {return Singleton.instance;}Singleton.instance = this;this.data = [];}addData(value)

4B参数秒杀GPT-3.5:MiniCPM 3.0惊艳登场!

​ 面壁智能 在 AI 的世界里,总有那么几个时刻让人惊叹不已。面壁智能推出的 MiniCPM 3.0,这个仅有4B参数的"小钢炮",正在以惊人的实力挑战着 GPT-3.5 这个曾经的AI巨人。 MiniCPM 3.0 MiniCPM 3.0 MiniCPM 3.0 目前的主要功能有: 长上下文功能:原生支持 32k 上下文长度,性能完美。我们引入了

Vue3项目开发——新闻发布管理系统(六)

文章目录 八、首页设计开发1、页面设计2、登录访问拦截实现3、用户基本信息显示①封装用户基本信息获取接口②用户基本信息存储③用户基本信息调用④用户基本信息动态渲染 4、退出功能实现①注册点击事件②添加退出功能③数据清理 5、代码下载 八、首页设计开发 登录成功后,系统就进入了首页。接下来,也就进行首页的开发了。 1、页面设计 系统页面主要分为三部分,左侧为系统的菜单栏,右侧

【即时通讯】轮询方式实现

技术栈 LayUI、jQuery实现前端效果。django4.2、django-ninja实现后端接口。 代码仓 - 后端 代码仓 - 前端 实现功能 首次访问页面并发送消息时需要设置昵称发送内容为空时要提示用户不能发送空消息前端定时获取消息,然后展示在页面上。 效果展示 首次发送需要设置昵称 发送消息与消息展示 提示用户不能发送空消息 后端接口 发送消息 DB = []@ro

模版方法模式template method

学习笔记,原文链接 https://refactoringguru.cn/design-patterns/template-method 超类中定义了一个算法的框架, 允许子类在不修改结构的情况下重写算法的特定步骤。 上层接口有默认实现的方法和子类需要自己实现的方法

easyui同时验证账户格式和ajax是否存在

accountName: {validator: function (value, param) {if (!/^[a-zA-Z][a-zA-Z0-9_]{3,15}$/i.test(value)) {$.fn.validatebox.defaults.rules.accountName.message = '账户名称不合法(字母开头,允许4-16字节,允许字母数字下划线)';return fal

【iOS】MVC模式

MVC模式 MVC模式MVC模式demo MVC模式 MVC模式全称为model(模型)view(视图)controller(控制器),他分为三个不同的层分别负责不同的职责。 View:该层用于存放视图,该层中我们可以对页面及控件进行布局。Model:模型一般都拥有很好的可复用性,在该层中,我们可以统一管理一些数据。Controlller:该层充当一个CPU的功能,即该应用程序

迭代器模式iterator

学习笔记,原文链接 https://refactoringguru.cn/design-patterns/iterator 不暴露集合底层表现形式 (列表、 栈和树等) 的情况下遍历集合中所有的元素