php实时通信,php实现实时通信

如果英文ok的话,可以直接看这篇文章

实现实时通信一般有两种方式:

socket或comet。socket是比较好的解决方案,问题在于不是所有的浏览器都兼容,服务器端实现起来也稍微有点麻烦。相比之下,comet(基于HTTP长连接的"服务器推”)实现起来更加方便,而且兼容所有的浏览器。所以这次就来说说comet的php实现。

comet也有好几种实现方式,如iframe, http long request,二者的区别可以参考这篇文章。本文主要探讨http long request实现实时通信。

先说说http长链接是怎么回事,通俗点讲就是服务器不是一收到请求就直接吐数据,而是在那憋啊憋,一直憋到憋不住了,才告诉你执行结果。

$count = 10;

for($i=0; $i

{

// do something ...sleep(2);

}

echo '憋死我了';

至于憋多长时间,就看具体应用了,如果憋太久的话,服务器资源的占用也会是个问题。

现在我们就要通过这种方法来实现实时通信(其实是准实时),先说一下原理:

客户端发起一个ajax长链接查询,然后服务端就开始执行代码,主要是检查某个文件是否被更新,如果没有,睡一会(sleep),醒来接着检查

如果客户端又发起了一个查询链接(正常请求),服务端收到后,处理请求,处理完毕后更新某个特定文件的modify time

这时第一次ajax查询的后台代码还在执行,发现某个文件被更新,说明来了新请求,输出对应的结果

第一次ajax查询的callback被触发,更新页面,然后再发起一个新的ajax长链接

实战

客户端

Comet Test

publish customAlert

publish customAlert2

//console.log(data);}).subscribe('customAlert2', function(data){

console.log('customAlert2');

//console.log(data);});

$(document).ready(function() {

$("a.customAlert").click(function(event) {

NovComet.publish('customAlert');

});

$("a.customAlert2").click(function(event) {

NovComet.publish('customAlert2');

});

NovComet.run();

});

这段代码说的是,有个NovComet的Object,注册了customAlert和customAlert2事件,当页面载入完成时,对两个按钮又加了监听事件,当点击时NovComet会发布customAlert或customAlert2事件,然后NovComet执行了run方法。

NovComet

//NovComet.jsNovComet = {

sleepTime: 1000,

_subscribed: {},

_timeout: undefined,

_baseurl: "comet.php",

_args: '',

_urlParam: 'subscribed',

subscribe: function(id, callback) {

NovComet._subscribed[id] = {

cbk: callback,

timestamp: NovComet._getCurrentTimestamp()

};

return NovComet;

},

_refresh: function() {

NovComet._timeout = setTimeout(function() {

NovComet.run()

}, NovComet.sleepTime);

},

init: function(baseurl) {

if (baseurl!=undefined) {

NovComet._baseurl = baseurl;

}

},

_getCurrentTimestamp: function() {

return Math.round(new Date().getTime() / 1000);

},

run: function() {

var cometCheckUrl = NovComet._baseurl + '?' + NovComet._args;

for (var id in NovComet._subscribed) {

var currentTimestamp = NovComet._subscribed[id]['timestamp'];

cometCheckUrl += '&' + NovComet._urlParam+ '[' + id + ']=' +

currentTimestamp;

}

cometCheckUrl += '&' + NovComet._getCurrentTimestamp();

$.getJSON(cometCheckUrl, function(data){

switch(data.s) {

case 0: // sin cambios NovComet._refresh();

break;

case 1: // trigger for (var id in data['k']) {

NovComet._subscribed[id]['timestamp'] = data['k'][id];

NovComet._subscribed[id].cbk(data.k);

}

NovComet._refresh();

break;

}

});

},

publish: function(id) {

var cometPublishUrl = NovComet._baseurl + '?' + NovComet._args;

cometPublishUrl += '&publish=' + id;

$.getJSON(cometPublishUrl);

}

};

NovComet的run方法首先把之前注册的几个事件串成一个url,并且很狡猾地使用了”[]",类似:

?subscribed[customAlert]=1300016814&subscribed[customAlert2]=1300016814&1300016825,这样php收到后,就会得到$_GET[subscribed]数组,最后那个时间戳是为了避免请求被缓存。如果收到后台传过来的数据data的s值为0,说明什么也没发生,隔1秒后继续执行;如果data.s的值为1,说明NovComet的publish事件被触发,则调用对应的callback。

publish方法执行后,会构造一个类似: ?publish=customAlert 这样一个url发送到后台。后台检测到pubish参数,则获取该参数的值,并更新对应文件的mtime。

服务端

// comet.phpinclude('NovComet.php');

$comet = new NovComet();

$publish = filter_input(INPUT_GET, 'publish', FILTER_SANITIZE_STRING);

if ($publish != '') {

echo $comet->publish($publish);

} else {

foreach (filter_var_array($_GET['subscribed'], FILTER_SANITIZE_NUMBER_INT) as $key => $value) {

$comet->setVar($key, $value);

}

echo $comet->run();

}

如果收到publish参数,直接输出,否则执行run方法,至于run是怎么回事,且看下码。

// NovComet.phpclass NovComet {

const COMET_OK = 0;

const COMET_CHANGED = 1;

private $_tries;

private $_var;

private $_sleep;

private $_ids = array();

private $_callback = null;

public function __construct($tries = 20, $sleep = 2)

{

$this->_tries = $tries;

$this->_sleep = $sleep;

}

public function setVar($key, $value)

{

$this->_vars[$key] = $value;

}

public function setTries($tries)

{

$this->_tries = $tries;

}

public function setSleepTime($sleep)

{

$this->_sleep = $sleep;

}

public function setCallbackCheck($callback)

{

$this->_callback = $callback;

}

const DEFAULT_COMET_PATH = "/dev/shm/%s.comet";

public function run() {

if (is_null($this->_callback)) {

$defaultCometPAth = self::DEFAULT_COMET_PATH;

$callback = function($id) use ($defaultCometPAth) {

$cometFile = sprintf($defaultCometPAth, $id);

return (is_file($cometFile)) ? filemtime($cometFile) : 0;

};

} else {

$callback = $this->_callback;

}

for ($i = 0; $i < $this->_tries; $i++) {

foreach ($this->_vars as $id => $timestamp) {

if ((integer) $timestamp == 0) {

$timestamp = time();

}

$fileTimestamp = $callback($id);

if ($fileTimestamp > $timestamp) {

$out[$id] = $fileTimestamp;

}

clearstatcache();

}

if (count($out) > 0) {

return json_encode(array('s' => self::COMET_CHANGED, 'k' => $out));

}

sleep($this->_sleep);

}

return json_encode(array('s' => self::COMET_OK));

}

public function publish($id)

{

return json_encode(touch(sprintf(self::DEFAULT_COMET_PATH, $id)));

}

}

可以看到publish时,创建了一个以$id命名的文件。run时,如果发现该$id文件存在,且时间戳大于之前保存的该$id对应的时间戳(通过setVar设置的),说明$id事件被触发,处理完后把$id放到$out数组中,然后判断一下$out数组是否为空,如果不为空,则输出一段json。

如果一段时间内都没有触发事件(for循环执行完毕),也输出一段json,告诉前端执行完了,但是没有任何新情况。

说明

可以在客户端监听/发布多个事件

监听事件时,可以传一个callback,这样收到数据时就会出发该callback

当监听事件时,会传一个时间戳

当事件被publish时,会向后台发一个请求,并传递一个新的时间戳

服务端不会一直执行,如果指定时间内,没有任何请求被触发,则结束运行

客户端会重复上述过程(setTimeout & NovComet.run())

最后来一张图说明一下

运行一段时间后,没有收到任何publish事件,服务端结束执行

服务端返回一段json

客户端触发了一个事件,服务端收到事件,返回一段新的json

callback被触发

客户端进入下一次的ajax长链接查询

程序员灯塔

转载请注明原文链接:php实现实时通信

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
PHP写得及时聊天工具,结构设计的很好,有源码。是学习PHP的好资料 JeCat-Jabber(简称:JJ) 是一款完全由 PHP 开发的即时通讯软件。JJ 采用 Gtk2 图像界面库来实现用户界面。顾名思义 JJ 使用 Jabber 协议(XMPP) ,因此可以 和 包括 GTalk 在内的 其它任何 XMPP 即时通讯软件 聊天。 JJ 项目的初衷在于 演示 桌面窗口 和 PHP 多任务处理,这两大 在长期在 PHP程序员 眼中 几乎是“不可能”的任务。 同时 也是 PHP框架 JCAT 的演示项目,JCAT 是一款同时 支持 Web 和 桌面 的 PHP框架,按照计划 将在稍后 发布。 JJ 的特点 > 跨平台。JJ 在Linxu 环境下 开发,在 Windows 平台上也一样可以正常使用。 > 实用 标准的 XMPP协议,能够与其它的 Jabber 软件互相通讯,例如 Google GTalk、Pidgin、PSI、Spark、Pandion,以及其它的 Jabber网页聊天窗口 > 可更换界面皮肤,皮肤样式文件 采用 类似 CSS 的语法,便于美工独立工作。 > 纯 PHP 实现,从 通讯协议 到 图形界面,全部都由 PHP 开发,如果你正好是一名 PHP 程序员,你可以驾轻就熟地在 JJ 之上进行二次开发。 > 可整合到你的网站中。JJ 近期的完善 会使 JJ 更容易地 为你的网站所有,让你的网站 也可以有像 淘宝旺旺 那样的 专用聊天工具 [ PHP 图像界面 ] JJ 采用 Gtk图像界面库,Gtk库 中 提供了丰富的 图形界面窗体(Widget),以及灵活方便的 窗体布局方式。 Gtk 本身 被广泛应用在 Linux 平台上,Linux 最著名 的桌面环境之一 Gnome 即是众多 Gtk软件 的集中展示。 PHP-Gtk2 将 Gtk 窗体绑定到 PHP 语言中,允许程序员 通过 PHP语言 来创建、显示、销毁 Gtk 窗体。PHP-Gtk 项目由 PHP创始人 亲自负责,并且自项目创立一来 一直持续发展,目前已经支持 最新的 Gtk2。 [ PHP 多任务处理 ] PHP 直到 5.2都没有提供 稳定可靠的 多线程特性。 但 PHP 仍然有能力进行多任务处理。 多线程 因为涉及到 线程安全、线程同步 等 困难,在 支持线程的语言中,也常常建议尽量避免使用。 Linux 的多路复用 是 更稳定和安全的 多任务模式,从 PHP5.0 开始 多路复用 支持 Windows 平台。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值