您的位置:首页 > 其它

COMET服务器推技术 – 实现Web服务器“主动”向客户端发送数据

2012-07-27 15:19 411 查看
在WEB开发中常常遇到一种需要即时更新内容的情况,比如在线聊天室,基于Web的IM系统或者股票查看系统等等。这些系统无一例外地要求内容更新的及时性。即每次有了新的内容,都必须即时发送给客户端。由于B/S架构的先天特性,即HTTP协议是一种无状态无链接协议,所以要实现服务器端主动发送数据给客户端,传统方法是难以实现的。为了解决这一问题,COMET服务器推技术便应运而生。

在传统的解决方案中,对页面进行全部或者局部刷新,似乎是解决这一问题的唯一办法。早期基于Web的聊天室一般都采用这种方法。即在页面中插入一个隐藏的iframe,通过这个iframe不断地自动刷新来轮询服务器端以获得最新消息,亦或是采用AJAX技术,每相隔一段时间发起一次HTTP请求来更新内容。但是这种方法缺点是非常明显的。首先,延迟无法避免,没有办法做到完全的及时性。如果我们设定轮询间隔为5s,那么内容更新的最大延迟就会说5s。其次,为了追求及时性,频繁的刷新、轮询,会造成过大的服务器压力。当在线人数很多时,这种方法几乎就是变相的分布式拒绝服务攻击。

那么有没有一种更加划算的方法呢?当然有的,比如使用activeX控件或者JAVA Applet等实现Socket通信。不过这种方法需要另外开端口,在网络情况复杂特别是存在防火墙的情况下,会造成通信失败。另外,使用Socket通信,还对服务器存在一定的要求,需要自己实现一套C/S模式的东西,这不符合Web开发的初衷。

在HTTP中有一种长连接技术,可以模仿Socket通信实现服务器端主动向客户端浏览器发送数据。它的原理其实很简单:当服务器端接到客户端的询问请求后,将整个HTTP连接置于阻塞状态,即什么也不做,也不发送数据,也不关掉连接。直到客户端需要将最新数据返回给客户端时,将数据通过这个HTTP连接返回回去,并且关闭连接。这样,客户端看到的结果,就似乎是服务器端主动向客户端浏览器发送数据了。但是,关掉连接以后怎么办呢?这时候,可以通过客户端JS代码中的定时器,再次发起请求。这样,只有内容发生了变化,才会进行一次HTTP会话,所以整体效率比轮询方式要高出很多,同时还有了更好的及时性。

在人人网中,页面中的即时消息提醒、在线IM就是通过这种方法实现的。另外,在WebQQ等基于WEB的IM中也广泛使用了这种技术。当然,在HTML5中,提供了专用的持久连接套接字,能够实现真正的服务器主动发送数据给客户端。

那么如何用代码来实现这个COMET服务器推送呢?请看下面。这部分代码是我从网上收集过来的,基于PHP 和prototype库写成。详情参考http://www.blogjava.net/JAVA-HE/archive/2009/04/13/265249.html

PHP服务器端代码:

[codesyntax lang="php"]

$filename = dirname(__FILE__).’/data.txt’;

$msg = isset($_GET['msg']) ? $_GET['msg'] : ”;

if ($msg != ”)

{

//写入内容至文件

file_put_contents($filename,$msg);

die();

}

set_time_limit(0);

$lastmodif = isset($_GET['timestamp']) ? $_GET['timestamp'] : 0;

//取得文件最后修改时间

$currentmodif = filemtime($filename);

while ($currentmodif <= $lastmodif)

{

//有释放CPU占用率的作用

usleep(10000);

//清除文件缓存信息

clearstatcache();

$currentmodif = filemtime($filename);

}

// return a json array

$response = array();

$response['msg'] = file_get_contents($filename);

$response['timestamp'] = $currentmodif;

echo json_encode($response);

ob_flush();

flush();

?>

[/codesyntax]

JS客户端代码:

[codesyntax lang="javascript"]

/*****************************************

* @Description : Comet TEST

* @FileName : comet.js

* @Author : He Chang Min

* @Date : 2009-03-05

* @Comment :

******************************************/

var WebApp =

{

//程序入口函数

WebMain : function()

{

var ajax = new Ajax.Request(WebApp._url_,

{

method: ‘get’,

parameters: { ‘timestamp’ : WebApp._timestamp_ },

onSuccess: function(transport)

{

var response = transport.responseText.evalJSON();

WebApp._timestamp_ = response['timestamp'];

WebApp.handleResponse(response);

WebApp._noerror_ = true;

},

onComplete: function(transport)

{

if (!WebApp._noerror_)

{

setTimeout(WebApp.WebMain, 5000);

}else

{

setTimeout(WebApp.WebMain, 10);

}

WebApp._noerror_ = false;

}

});

},

handleResponse : function(response)

{

$(‘content’).innerHTML += ‘
‘ + response['msg'] + ‘

‘;

},

doRequest : function(request)

{

new Ajax.Request(WebApp._url_,

{

method : ‘get’,

parameters : { ‘msg’ : request }

});

},

//成员属性

_timestamp_ : 0,

_url_ : ‘./comet.php’,

_noerror_ : true

}

[/codesyntax]

以下是代码的打包文件:下载文件:
1277351865_4251418e.rar

可以将代码上传到服务器上,然后打开两个不同的浏览器。从一方发送数据,可以看到另一方会立即显示出更新。

以下是本文的参考资料,想要了解更多的朋友可以前去访问:

浅谈comet技术 http://www.blogjava.net/JAVA-HE/archive/2009/04/13/265249.html

Comet:基于 HTTP 长连接的“服务器推”技术 http://www.ibm.com/developerworks/cn/web/wa-lo-comet/

实战 Comet 应用程序开发 http://www.ibm.com/developerworks/cn/web/wa-lo-w2fpak-comet/index.html

javaeye上的Comet相关文章 http://www.javaeye.com/wiki/topic/684909
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: 
相关文章推荐