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

comet也有好几种实现方式,如iframe, http long request,本文主要探讨http long request实现实时通信。

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

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

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

1. 客户端发起一个ajax长链接查询,然后服务端就开始执行代码,主要是检查某个文件是否被更新,如果没有,睡一会(sleep),醒来接着检查
2. 如果客户端又发起了一个查询链接(正常请求),服务端收到后,处理请求,处理完毕后更新某个特定文件的modify time
3. 这时第一次ajax查询的后台代码还在执行,发现某个文件被更新,说明来了新请求,输出对应的结果
4. 第一次ajax查询的callback被触发,更新页面,然后再发起一个新的ajax长链接

 <?php //NovComet.php class 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)));}
}<?php //comet.php include('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(); }functionsend(msg){$.ajax({data: {'msg' : msg}, type : 'post', url : '{:U('Live/SendMsg')}', success : function(response){ //alert(response);;  }
})}$(document).ready(function(){connect();$("#btn").click(function(){ var msg = $('#msg').val();send(msg);msg.html('');});})public function SendMsg(){ $filename = './Uploads/live/'.'data.json'; if ($_POST['msg']!='') { file_put_contents($filename,$_POST['msg']); $this->ajaxReturn($_POST,'OK',100); die();}else{ $this->ajaxReturn($_POST,'on',0); die();}}<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.1//EN" "http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd"> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <title>Comet demo</title> <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> <script type="text/javascript" src="./jquery-1.8.2.min.js"></script> <script type="text/javascript" src="./json2.js"></script> <script> var timestamp = 0; var url = 'backend.php'; var error = false; //通过ajax建立和php端处理函数的连接(通过递归调用建立长时间的连接) function connect(){$.ajax({data: {'timestamp' : timestamp},url: url,type: 'get',timeout: 0,success: function(response){ var data = JSON.parse(response);error= false;timestamp= data.timestamp; if (data.msg != undefined && data.msg != ""){$("#content").append("<div>" + data.msg + "</div>");}},error: function(){error= true;setTimeout(function(){ connect();}, 5000);},complete: function(){ if (error) //请求有错误时,延迟5s再连接  setTimeout(function(){connect();}, 5000); else connect();
}})}//发送信息 function send(msg){$.ajax({data: {'msg' : msg},type: 'get',url:url})}//创建长时间的连接  $(document).ready(function(){
connect();})</script> </head> <body> <div id="content"></div> <form action="" method="get" οnsubmit="send($('#word').val());$('#word').val('');return false;"> <input type="text" name="word" id="word" value="" /> <input type="submit" name="submit" value="Send" /> </form> </body> </html><?php //设置请求运行时间不限制,解决因为超过服务器运行时间而结束请求 ini_set("max_execution_time", "0"); $filename = dirname(__FILE__).'/data.txt'; $msg = isset($_GET['msg']) ? $_GET['msg'] : ''; // 判断页面提交过来的修改内容是否为空,不为空则将内容写入文件,并中断流程 if ($msg != '')
{ file_put_contents($filename,$msg); exit;
}/*获取文件上次修改时间戳 和 当前获取到的最近一次文件修改时间戳* 文件上次修改时间戳 初始 默认值为0* 最近一次文件修改时间戳 通过 函数 filemtime()获取*/ $lastmodif = isset($_GET['timestamp']) ? $_GET['timestamp'] : 0; clearstatcache(); //清除文件状态缓存 $currentmodif = filemtime($filename); /* 如果当前返回的文件修改unix时间戳小于或等于上次的修改时间,*表明文件没有更新不需要推送消息*如果当前返回的文件修改unix时间戳大于上次的修改时间* 表明文件有更新需要输出修改的内容作为推送消息 */ while ($currentmodif <= $lastmodif)
{usleep(10000); //休眠10ms释放cpu的占用 clearstatcache(); // 清除文件状态缓存 $currentmodif = filemtime($filename);
} //推送信息处理(需要推送说明文件有更改,推送信息包含本次修改时间、内容) $response = array(); $response['msg'] = file_get_contents($filename); $response['timestamp'] = $currentmodif; echo json_encode($response); flush(); ?>

转载于:https://www.cnblogs.com/520fyl/p/5396317.html

PHP 实现实时通信一般有两种方式相关推荐

  1. 继承WebMvcConfigurer 和 WebMvcConfigurerAdapter类依然CORS报错? springboot 两种方式稳定解决跨域问题

    继承WebMvcConfigurer 和 WebMvcConfigurerAdapter类依然CORS报错???springboot 两种方式稳定解决跨域问题! 之前我写了一篇文章,来解决CORS报错 ...

  2. 周一02.3运行python程序的两种方式

    一.运行python程序的两种方式 方法一:交互式:                      优点:输入一行代码立刻返回结果                       缺点:无法永久保存代码 方法 ...

  3. 实验四:使用库函数API和C代码中嵌入汇编代码两种方式使用同一个系统调用

    贺邦+原创作品转载请注明出处 + <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 实验目的: 使用库函数 ...

  4. android不调用系统发送短信,android之两种方式调用短信发送接口

    释放双眼,带上耳机,听听看~! 相信很多程序员在开发程序的时候都会遇到短信调用端口的情况,今天是技术狗小编为大家带来的关于android之两种方式调用短信发送接口,希望对你学习这方面知识有帮助! an ...

  5. android asynctask源码分析,Android通过Handler与AsyncTask两种方式动态更新ListView(附源码)...

    本文实例讲述了Android通过Handler与AsyncTask两种方式动态更新ListView的方法.分享给大家供大家参考,具体如下: 有时候我们需要修改已经生成的列表,添加或者修改数据,noti ...

  6. VS Code - Debugger for Chrome调试JavaScript的两种方式

    VS Code - Debugger for Chrome调试JavaScript的两种方式 最近由于出差的缘故,博客写的不是很多,一直想写一篇VS Code - Debugger for Chrom ...

  7. 简单介绍python连接telnet和ssh的两种方式

    本文主要介绍了python连接telnet和ssh的两种方式,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下 Telnet 连接方式 #!/usr/bin/env p ...

  8. 简单介绍MySQL开启事务的两种方式

    本篇文章给大家分享MySQL 是如何开启一个事务的,原文通过两种方式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧 方式 1 START TRANSACTION 或 ...

  9. 两种方式(goto语句以及while循环)实现C语言关机小程序

    用C语言写的关机小程序简单易懂,可以拿去整蛊室友同学(每天一个挨打小技巧),下面我就给出两种方式实现这个代码,欢迎大家讨论补充. 1.用goto语句实现: #include <stdio.h&g ...

最新文章

  1. 【机器视觉案例】(5) AI视觉,远程手势控制虚拟计算器,附python完整代码
  2. C/C++中switch用法的一种替换方式
  3. UIGestureRecognizer手势介绍
  4. 在Windows上编译最新版本的Hashcat
  5. java idea 模块_IDEA搭建java多模块module项目-Go语言中文社区
  6. Win11提示无法安装程序怎么办 Win11提示无法安装程序的解决方法
  7. 怎么设置mysql 的权限_怎么设置SQL数据库用户权限
  8. 使用Git子模块和开发模式管理Python项目
  9. RHCE 学习笔记(32) - DNS
  10. office2003安装包下载,专业版完整版官方原版!
  11. 单日峰值2T发送量邮件营销平台实践经验
  12. 基于SVG的绘制多边形jQuery插件
  13. 快速学习-帕特里夏树
  14. C++:实现一些简单的方法来 布莱克-斯科尔斯期权估值理论(附完整源码)
  15. 从一名白纸交易者到稳定盈利交易员需要多长时间?
  16. HTML5+CSS3小实例:酷炫的菱形加载动画
  17. python设计一个动物类_「Python」每日一练:设计圆类计算周长和面积、设计动物类...
  18. word标尺灰色_如何在Microsoft Word中显示和隐藏标尺
  19. 基于VS2017+OpenCV,C++搭建NanoDet-Plus轻量级目标检测模型并训练(二)
  20. c++学习笔记- 自制魔兽显血改键

热门文章

  1. PHP文件操作【其一】文件路径
  2. 新车入手,美利达公爵500
  3. 系统集成资质-谈项管论文备考写作:论文写作几大误区
  4. 如何查看系统启动时间-转
  5. 计算机科学论文吧,计算机科学毕业论文.doc
  6. [Java] 蓝桥杯ADV-210 算法提高 2-1屏幕打印
  7. [Java] 蓝桥杯BASIC-29 基础练习 高精度加法
  8. L1-005. 考试座位号-PAT团体程序设计天梯赛GPLT
  9. oracle如何调试sql,调试oracle与调试sql server存储过程
  10. java项目学习_一个Java项目的学习