成人午夜视频全免费观看高清-秋霞福利视频一区二区三区-国产精品久久久久电影小说-亚洲不卡区三一区三区一区

PHP長連接實現(xiàn)與使用方法的示例分析-創(chuàng)新互聯(lián)

這篇文章給大家分享的是有關(guān)PHP長連接實現(xiàn)與使用方法的示例分析的內(nèi)容。小編覺得挺實用的,因此分享給大家做個參考,一起跟隨小編過來看看吧。

創(chuàng)新互聯(lián)專注于企業(yè)營銷型網(wǎng)站建設(shè)、網(wǎng)站重做改版、大渡口網(wǎng)站定制設(shè)計、自適應(yīng)品牌網(wǎng)站建設(shè)、H5場景定制、成都商城網(wǎng)站開發(fā)、集團公司官網(wǎng)建設(shè)、外貿(mào)網(wǎng)站制作、高端網(wǎng)站制作、響應(yīng)式網(wǎng)頁設(shè)計等建站業(yè)務(wù),價格優(yōu)惠性價比高,為大渡口等各大城市提供網(wǎng)站開發(fā)制作服務(wù)。

本文實例講述了PHP長連接實現(xiàn)與使用方法。分享給大家供大家參考,具體如下:

長連接技術(shù)(Long Polling)

在服務(wù)器端hold住一個連接, 不立即返回, 直到有數(shù)據(jù)才返回, 這就是長連接技術(shù)的原理

長連接技術(shù)的關(guān)鍵在于hold住一個HTTP請求, 直到有新數(shù)據(jù)時才響應(yīng)請求, 然后客戶端再次自動發(fā)起長連接請求.

那怎么樣hold住一個請求呢?服務(wù)器端的代碼可能看起來像這樣的

set_time_limit(0); //這句很重要, 不至于運行超時
while (true) {
  if (hasNewMessage()) {
    echo json_encode(getNewMessage());
    break;
  }
  usleep(100000);   //避免太過頻繁的查詢
}

沒錯,就是通過循環(huán)來實現(xiàn)hold住一個請求, 不至于立即返回. 查詢到有新數(shù)據(jù)之后才響應(yīng)請求. 然后客戶端處理數(shù)據(jù)后,再次發(fā)起長連接請求.

客戶端的代碼是像這樣的

<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>

一個簡易的聊天室

通過長連接, 我們可以開發(fā)一個簡易的web聊天室

下面, 我們通過redis開發(fā)一個簡易的web聊天室

1. 每一個客戶端發(fā)起長連接時, 在服務(wù)器端生成一個消息隊列, 對應(yīng)該用戶. 然后監(jiān)聽有無新數(shù)據(jù), 有則返回數(shù)據(jù)到客戶端進行處理, 并再起發(fā)起長連接請求.

2. 每一個客戶端發(fā)起消息時, 進行消息隊列的廣播.

下面是代碼片段:

<?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);
    }
  }
}

長連接避免了過于頻繁的輪詢. 但服務(wù)器維持一個長連接也有額外的資源消耗. 大并發(fā)時性能不理想. 在小型應(yīng)用里面可以考慮使用

更建議客戶端使用html5的websocket協(xié)議, 服務(wù)器端使用swoole.

感謝各位的閱讀!關(guān)于“PHP長連接實現(xiàn)與使用方法的示例分析”這篇文章就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,讓大家可以學到更多知識,如果覺得文章不錯,可以把它分享出去讓更多的人看到吧!

分享標題:PHP長連接實現(xiàn)與使用方法的示例分析-創(chuàng)新互聯(lián)
瀏覽地址:http://jinyejixie.com/article24/ccspje.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站營銷、企業(yè)建站標簽優(yōu)化、服務(wù)器托管微信公眾號、網(wǎng)站排名

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

成都定制網(wǎng)站建設(shè)
襄城县| 鹤壁市| 嘉定区| 龙门县| 寿光市| 叶城县| 高邮市| 子长县| 肇源县| 新巴尔虎右旗| 乌鲁木齐市| 定州市| 漳浦县| 北流市| 长垣县| 荣成市| 莲花县| 宝坻区| 元谋县| 合作市| 桑日县| 濮阳市| 方正县| 乌鲁木齐市| 镇巴县| 吉安县| 巴青县| 泰顺县| 盐津县| 德清县| 廉江市| 宽甸| 德江县| 黎城县| 广饶县| 泽库县| 娱乐| 温州市| 邳州市| 太湖县| 乌海市|