PHP长连接实现与使用方法详解
发布:smiling 来源: PHP粉丝网 添加日期:2021-09-03 11:34:34 浏览: 评论:0
这篇文章主要介绍了PHP长连接实现与使用方法,结合实例形式较为详细的分析了php长连接的概念、功能、实现与使用方法,需要的朋友可以参考下。
本文实例讲述了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/
Tags: PHP长连接
- 上一篇:搜索附近的人PHP实现代码
- 下一篇:PHP使用Redis长连接的方法详解
相关文章
- ·浅析关于PHP中Sphinx长连接问题(2014-08-27)
- ·PHP使用Redis长连接的方法详解(2021-09-03)
推荐文章
热门文章
最新评论文章
- 写给考虑创业的年轻程序员(10)
- PHP新手上路(一)(7)
- 惹恼程序员的十件事(5)
- PHP邮件发送例子,已测试成功(5)
- 致初学者:PHP比ASP优秀的七个理由(4)
- PHP会被淘汰吗?(4)
- PHP新手上路(四)(4)
- 如何去学习PHP?(2)
- 简单入门级php分页代码(2)
- php中邮箱email 电话等格式的验证(2)