ホームページ >バックエンド開発 >PHPチュートリアル >PHP 長時間接続のユースケース分析

PHP 長時間接続のユースケース分析

php中世界最好的语言
php中世界最好的语言オリジナル
2018-05-18 14:48:492239ブラウズ

今回は、PHP の長い接続を使用する場合の事例分析をお届けします。PHP の長い接続を使用する際の 注意事項 は何ですか?実際の事例を見てみましょう。

ロングポーリング

サーバー側で接続を保持し、データがあるまですぐに戻らないというのが、ロング接続技術の原理です

ロング接続技術の鍵は、HTTPリクエストを保持することです。新しいデータが存在するまでリクエストに応答しません。その後、クライアントは再び長い接続リクエストを自動的に開始します

それでは、リクエストを保留するにはどうすればよいでしょうか?サーバー側のコードは次のようになります

set_time_limit(0); //这句很重要, 不至于运行超时
while (true) {
  if (hasNewMessage()) {
    echo json_encode(getNewMessage());
    break;
  }
  usleep(100000);   //避免太过频繁的查询
}
はい、リクエストはループを通じて保留され、新しいデータが見つかった後、クライアントがデータを処理した後にのみ応答されます。 、再び長いリクエストが開始されます。接続リクエストです

クライアントコードは次のとおりです

<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);
    }
  }
}
長い接続は頻繁なポーリングを回避します。ただし、サーバーには、長時間の接続を維持するための追加のリソースも必要です。小規模なアプリケーションでの使用は、理想的ではありません。サーバー側では swoole を使用します

この記事の事例などを読んで方法を習得したと思います。php 中国語 Web サイトの他の関連記事にも注目してください。

推奨読書:

PHP名前空間の使用の詳細な説明

PHP中国語ツールクラス ChineseUtil を使用して中国語の文字とピンインを変換する方法

以上がPHP 長時間接続のユースケース分析の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。