PHP长连接实现与使用方法详解


Posted in PHP onFebruary 11, 2018

本文实例讲述了PHP长连接实现与使用方法。分享给大家供大家参考,具体如下:

长连接技术(Long Polling)

在服务器端hold住一个连接, 不立即返回, 直到有数据才返回, 这就是长连接技术的原理

长连接技术的关键在于hold住一个HTTP请求, 直到有新数据时才响应请求, 然后客户端再次自动发起长连接请求.

那怎么样hold住一个请求呢?服务器端的代码可能看起来像这样的

set_time_limit(0); //这句很重要, 不至于运行超时
while (true) {
  if (hasNewMessage()) {
    echo json_encode(getNewMessage());
    break;
  }
  usleep(100000);   //避免太过频繁的查询
}

没错,就是通过循环来实现hold住一个请求, 不至于立即返回. 查询到有新数据之后才响应请求. 然后客户端处理数据后,再次发起长连接请求.

客户端的代码是像这样的

<script type="text/javascript">
  (function longPolling() {
    $.ajax({
      'url': 'server.php',
      'data': data,
      'dataType': 'json',
      'success': function(data) {
        processData(data);
        longPolling();
      },
      'error': function(data) {
        longPolling();
      }
    });
  })();
</script>

一个简易的聊天室

通过长连接, 我们可以开发一个简易的web聊天室

下面, 我们通过redis开发一个简易的web聊天室

1. 每一个客户端发起长连接时, 在服务器端生成一个消息队列, 对应该用户. 然后监听有无新数据, 有则返回数据到客户端进行处理, 并再起发起长连接请求.

2. 每一个客户端发起消息时, 进行消息队列的广播.

下面是代码片段:

<?php
namespace church\LongPolling;
use Closure;
use church\LongPolling\Queue\RedisQueue;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpFoundation\JsonResponse;
class Server
{
  public $event = [];
  public $redisQueue = null;
  public $request = null;
  public $response = null;
  public function __construct()
  {
    $this->redisQueue = new RedisQueue();
    $this->request = Request::createFromGlobals();
    $this->response = new JsonResponse();
  }
  public function on($event, Closure $closure)
  {
    if (is_callable($closure)) {
      $this->event[$event][] = $closure;
    }
  }
  public function fire($event)
  {
    if (isset($this->event[$event])) {
      foreach ($this->event[$event] as $callback) {
        call_user_func($callback, $this);
      }
    }
  }
  public function sendMessage($data)
  {
    switch ($data['type']) {
      case 'unicast':   //单播
        $this->unicast($data['target'], $data['data'], $data['resource']);
        break;
      case 'multicast':    //组播
        foreach ($data['target'] as $target) {
          $this->unicast($target, $data['data'], $data['resource']);
        }
        break;
      case 'broadcast':    //广播
        foreach ($this->redisQueue->setQueueName('connections') as $target) {
          $this->unicast($target, $data['data'], $data['resource']);
        }
        break;
    }
    $this->fire('message');
  }
  public function unicast($target, $message, $resource = 'system')
  {
    $redis_queue = new RedisQueue();
    $redis_queue->setQueueName($target)->push($resource . ':' . $message);
  }
  public function getMessage($target)
  {
    return $this->redisQueue->setQueueName($target)->pop();
  }
  public function hasMessage($target)
  {
    return count($this->redisQueue->setQueueName($target));
  }
  public function run()
  {
    $data = $this->request->request;
    while (true) {
      if ($data->get('action') == 'getMessage') {
        if ($this->hasMessage($data->get('target'))) {
          $this->response->setData([
            'state' => 'ok',
            'message' => '获取成功',
            'data' => $this->getMessage($data->get('target'))
          ]);
          $this->response->send();
          break;
        }
      } elseif ($data->get('action') == 'connect') {
        $exist = false;
        foreach ($this->redisQueue->setQueueName('connections') as $connection) {
          if ($connection == $data->get('data')) {
            $exist = true;
          }
        }
        if (! $exist) {
          $this->redisQueue->setQueueName('connections')->push($data->get('data'));
        }
        $this->fire('connect');
        break;
      }
      usleep(100000);
    }
  }
}

长连接避免了过于频繁的轮询. 但服务器维持一个长连接也有额外的资源消耗. 大并发时性能不理想. 在小型应用里面可以考虑使用

更建议客户端使用html5的websocket协议, 服务器端使用swoole.

有关swoole, 你可以查看官网:https://www.swoole.com/

希望本文所述对大家PHP程序设计有所帮助。

PHP 相关文章推荐
对squid中refresh_pattern的一些理解和建议
Apr 17 PHP
php模拟post行为代码总结(POST方式不是绝对安全)
Feb 22 PHP
PHP获取和操作配置文件php.ini的几个函数介绍
Jun 24 PHP
PHP关于IE下的iframe跨域导致session丢失问题解决方法
Oct 10 PHP
session在php5.3中的变化 session_is_registered() is deprecated in
Nov 12 PHP
Thinkphp模板中截取字符串函数简介
Jun 17 PHP
php分割合并两个字符串的函数实例
Jun 19 PHP
搭建Vim为自定义的PHP开发工具的一些技巧
Dec 11 PHP
[原创]php简单隔行变色功能实现代码
Jul 09 PHP
理清PHP在Linxu下执行时的文件权限方法
Jun 07 PHP
PHP通过get方法获得form表单数据方法总结
Sep 12 PHP
Laravel 使用查询构造器配合原生sql语句查询的例子
Oct 12 PHP
搜索附近的人PHP实现代码
Feb 11 #PHP
PHP的RSA加密解密方法以及开发接口使用
Feb 11 #PHP
php+ajax实现无刷新文件上传功能(ajaxuploadfile)
Feb 11 #PHP
PHP实现的多维数组排序算法分析
Feb 10 #PHP
ThinkPHP整合datatables实现服务端分页的示例代码
Feb 10 #PHP
PHP实现APP微信支付的实例讲解
Feb 10 #PHP
PHP有序表查找之插值查找算法示例
Feb 10 #PHP
You might like
UCenter Home二次开发指南
2009/05/28 PHP
国产PHP开发框架myqee新手快速入门教程
2014/07/14 PHP
PHP中Notice错误常见解决方法
2017/04/28 PHP
PHP提取字符串中的手机号正则表达式怎么写
2017/07/17 PHP
php用户名的密码加密更安全的方法
2019/06/21 PHP
特殊字符、常规符号及其代码对照表
2006/06/26 Javascript
jquery 新手学习常见问题解决方法
2010/04/18 Javascript
菜鸟javascript基础整理1
2010/12/06 Javascript
UpdatePanel和Jquery冲突的解决方法
2013/04/01 Javascript
Js(JavaScript)中,弹出是或否的选择框示例(confirm用法的实例分析)
2013/07/09 Javascript
Iframe实现跨浏览器自适应高度解决方法
2014/09/02 Javascript
javascript函数式编程实例分析
2015/04/25 Javascript
js实现定时进度条完成后切换图片
2017/01/04 Javascript
关于vue-router路径计算问题
2017/05/10 Javascript
Vue中img的src属性绑定与static文件夹实例
2017/05/18 Javascript
Angular 2父子组件数据传递之@Input和@Output详解(下)
2017/07/05 Javascript
解决vue.js在编写过程中出现空格不规范报错的问题
2017/09/20 Javascript
详解JavaScript事件循环机制
2018/09/07 Javascript
基于Angular中ng-controller父子级嵌套的相关属性详解
2018/10/08 Javascript
理理Vue细节(推荐)
2019/04/16 Javascript
js 根据对象数组中的属性进行排序实现代码
2019/09/12 Javascript
[49:35]LGD vs OG 2018国际邀请赛淘汰赛BO3 第二场 8.25
2018/08/29 DOTA
Python入门篇之数字
2014/10/20 Python
详解Python函数作用域的LEGB顺序
2016/05/14 Python
把django中admin后台界面的英文修改为中文显示的方法
2019/07/26 Python
python try except返回异常的信息字符串代码实例
2019/08/15 Python
从Pytorch模型pth文件中读取参数成numpy矩阵的操作
2021/03/04 Python
阿迪达斯比利时官方商城:adidas比利时
2016/10/10 全球购物
英国时尚饰品和发饰购物网站:Claire’s
2017/07/04 全球购物
西班牙拥有最佳品牌的动物商店:Animalear.com
2018/01/05 全球购物
简述DNS进行域名解析的过程
2013/12/02 面试题
资产经营总监岗位职责
2013/12/04 职场文书
幼儿园优秀教师事迹
2014/02/13 职场文书
法制宣传标语
2014/06/23 职场文书
会议通知
2015/04/15 职场文书
Java 数组的使用
2022/05/11 Java/Android