ホームページ >バックエンド開発 >PHPチュートリアル >WebSocketを使ったphpの詳しい解説 example_PHPチュートリアル

WebSocketを使ったphpの詳しい解説 example_PHPチュートリアル

WBOY
WBOYオリジナル
2016-07-13 10:36:04839ブラウズ

以下に、クライアントとサーバーの間で WebSocket 接続を確立するときのハンドシェイク部分を示す図を描きました。これは、ノードによって提供されるネット モジュールが開発者によってすでにカプセル化されているため、この部分はノード内で非常に簡単に完了できます。接続の確立に対処する必要はなく、データの相互作用のみを考慮する必要があります。ただし、PHPはそうではなく、ソケットの接続、確立、バインディング、監視などを自分で操作する必要があるため、取り出して話す必要があります。

WebSocketを使ったphpの詳しい解説 example_PHPチュートリアル

①と②は実際にはHTTPリクエストとレスポンスですが、処理中に取得するのは解析されていない文字列です。例:

コードをコピーします コードは次のとおりです:

GET /chat HTTP/1.1
ホスト:server.example.com
オリジン:http://www.jb51.com

通常、このようなリクエストがサーバーに到達すると、コード ライブラリを通じてこの情報を直接取得できます。

1. phpでWebSocketを処理する

WebSocket 接続はクライアントによってアクティブに開始されるため、すべてはクライアントから開始する必要があります。最初のステップは、クライアントから送信された Sec-WebSocket-Key 文字列を解析することです。

コードをコピーします コードは次のとおりです:

GET /chat HTTP/1.1
ホスト:server.example.com
アップグレード:websocket
接続:アップグレード
Sec-WebSocket-Key: dGhlIHNhbXBsZSBub25jZQ ==
Origin: http://www.jb51.com
Sec-WebSocket-Protocol: チャット、スーパーチャット
Sec-WebSocket-Version: 13

クライアントリクエストフォーマット

まず、php はソケット接続を確立し、ポート情報をリッスンします。

1.ソケット接続の確立

ソケットの確立については、大学でコンピュータ ネットワークを学んだことのある多くの人が知っていると思います。次は、接続を確立するプロセスの図です。

コードをコピーします コードは次のとおりです:
// ソケットを作成します
$master =ソケット_create(AF_INET, SOCK_STREAM, SOL_TCP);
socket_set_option($master, SOL_SOCKET, SO_REUSEADDR, 1); socket_bind ($master, $address, $port);
socket_listen($master);


ここの処理はnodeに比べて面倒です。 上記のコード行は接続を確立しませんが、ソケットを確立するためにこれらのコードを書かなければなりません。処理プロセスが少し複雑なので、管理や呼び出しを容易にするために、さまざまな処理をクラスに記述しました。

コードをコピーします コードは次のとおりです:

//demo.php
Class WS {
    var $master;  // 连接 server 的 client
    var $sockets = array(); // 不同状态的 socket 管理
    var $handshake = false; // 判断是否握手

    function __construct($address, $port){
        // 建立一个 socket 套接字
        $this->master = socket_create(AF_INET, SOCK_STREAM, SOL_TCP)  
            or die("socket_create() failed");
        socket_set_option($this->master, SOL_SOCKET, SO_REUSEADDR, 1) 
            or die("socket_option() failed");
        socket_bind($this->master, $address, $port)                   
            or die("socket_bind() failed");
        socket_listen($this->master, 2)                              
            or die("socket_listen() failed");

        $this->sockets[] = $this->master;

        // debug
        echo("Master socket  : ".$this->master."\n");

        while(true) {
            //自动选择来消息的 socket 如果是握手 自动选择主机
            $write = NULL;
            $except = NULL;
            socket_select($this->sockets, $write, $except, NULL);

foreach ($this->sockets as $socket) {
//主机のクライアントに接続します
if ($socket == $this->master){
$client =ソケット_accept($this->master);
if ($client < 0) {
// debug
echo "socket_accept() が失敗しました";
continue;
} else {
//connect($client);
array_push($this->sockets, $ client);
echo "connect clientn";
}
} else {
$bytes = @socket_recv($s ocket,$buffer,2048,0);
if($bytes == 0) return;
if (!$ this->handshake) {
// 如果没有手,先手握回应
// doHandShake($socket, $buffer);
echo "shakeHandsn";
} else {
// 如果已经握り手、直接受信データ、并处理
$buffer = decode($buffer);
//process($socket, $buffer);
echo "ファイルを送信";
}
}
}
}
}
}

上のこのセグメントコードは私が承認済みであり、大した問題はありませんが、必要に応じて cmd 実行コマンドで php /path/to/demo.php にアクセスできます。当然、上だけが 1 つの種類であり、必要な場合は

新しい例も入手します。

复制代码代码如下:

$ws = new WS('localhost', 4000);

客户端代网可以稍微简单点:

复制代価代価如下:

var ws = new WebSocket("ws://localhost:4000");
ws.onopen = function(){
console.log("ハンドシェイク成功");
};
ws.onerror = function() {
console.log("error");
};

クライアントが接続すると、サーバー コードを実行します。

2. Sec-WebSocket-Key 情報を抽出します

コードをコピー コードは次のとおりです:

function getKey($req) {
$key = null;
if (preg_match("/Sec-WebSocket-Key: (.*)rn/") , $req , $match)) {

これは比較的単純で、直接的な正規のマッチングです。WebSocket 情報ヘッダーには Sec-WebSocket-Key が含まれている必要があるため、マッチングが高速になります~

3. 暗号化 Sec-WebSocket キー

コードをコピーします

コードは次のとおりです:

function encry($req){ $key = $this->getKey($req); $mask = "258EAFA5-E914-47DA -95CA-C5AB0DC85B11 ";

returnbase64_encode(sha1($key . '258EAFA5-E914-47DA-95CA-C5AB0DC85B11', true));
}


SHA-1暗号化文字列をbase64で再度暗号化します。暗号化アルゴリズムが間違っている場合、クライアントはチェック時にエラーを直接報告します:



4. Sec-WebSocket-Accept に返信します

コードをコピーします

コードは次のとおりです:

function dohandshake($socket, $req){ // 暗号化キーを取得します $acceptKey = $this->encry($req); $upgrade = " HTTP/1.1 101 スイッチング プロトコルrn" .
" "アップグレード: websocketrn" .
" "接続: Upgradern" .
" "Sec-WebSocket-Accept: " . $acKeycept . "rn" .
"rn ";

// ソケットに書き込みます
ソケット_write(socket,$upgrade.chr(0), strlen($upgrade.chr(0)));
// ハンドシェイクが成功したことをマークし、次回データが受け入れられるようにしますデータフレーム形式
$this ->handshake = true;
}



ここで必ず注意してください。すべてのリクエストと対応する形式には最後に空白行があります。これは、テストを開始したときに失われ、長い間苦労しました。


クライアントがキーのチェックに成功すると、onopen 関数がトリガーされます:

5. データフレームの処理

コードをコピー

コードは次のとおりです:

// データフレームを解析function decode($buffer) { $len = $masks = $data = $decoded = null ; $ len = ord($buffer[1]) & 127;
if ($len === 126) {
$masks = substr($buffer, 4, 4);
$data = substr($buffer, 8);
} else if ($len === 127) {
$masks = substr($buffer, 10, 4); $data = substr($buffer, 14);

}
for ($index = 0; $index < strlen($data) ); $index++) {
$decoded .= $data[$index] ^ $masks[$index % 4] ;
}
return $decoded;
}

ここで関係するエンコーディングの問題については、前の記事で述べたので、ここでは詳しく説明しません。PHP には文字処理に関する関数が多すぎて、詳しくは覚えていません。ここではデコードプログラムを作成し、クライアントに直接データを送信します。これはチャット ルームのモデルとみなすことができます。

コードをコピー コードは次のとおりです:

//フレーム情報処理を返す
function Frame($s) {
$a = str_split($s, 125);
if (count($a) ) == 1) {
= "x81" . $ns;
}

//データを返す
function send($client, $msg){
$msg = $this->frame($msg);
ソケット_write($client, $msg, strlen($msg));
}



クライアントコード:


コードをコピー

コードは次のとおりです:

var ws = new WebSocket("ws://localhost:4000");ws.onopen = function(){ コンソール。 log( "ハンドシェイクが成功しました");};
ws.onmessage = function(e){
console.log("message:" + e.data);
};
ws.onerror = function(){
console .log ("エラー");
};
ws.send("李京");



接続後にデータを送信すると、サーバーはそのまま戻ります:



2. 問題に注意してください

1.WebSocketのバージョンの問題

ハンドシェイク中のクライアントのリクエストには、バージョン識別子である Sec-WebSocket-Version: 13 が含まれています。これは、現在のすべてのブラウザーがこのバージョンを使用します。以前のバージョンは、データ暗号化部分でさらに面倒でした:

コードをコピーします

コードは次のとおりです:

GET /chat HTTP/1.1Host:server.example.com アップグレード: websocket接続: アップグレード
発信元: http://www.jb51.net
Sec-WebSocket-Protocol: チャット、スーパーチャット
Sec-WebSocket-Key1: xxxx
Sec-WebSocket-Key2: xxxx



このバージョン(古くてもう使用されていない)の場合は、次の方法で入手する必要があります


コードをコピーします

コードは次のとおりです:

function encry($key1,$key2 ,$l8b){ //数値を取得します preg_match_all('/([d]+)/', $key1, $key1_num); preg_match_all('/([d]+)/', $key2, $key2_num); $key1_num = implode($key1_num[0]);$key2_num = implode($key2_num[0]);
//スペースをカウント
preg_match_all('/([ ]+)/', $key1, $key1_spc) ;
preg_match_all('/([ ]+)/', $key2, $key2_spc);

if($key1_spc==0|$key2_spc==0){ $this->log("無効なキー");return; }
//いくつかの計算
$key1_sec = Pack("N",$key1_num / $key1_spc);
$key2_sec = Pack("N",$key2_num / $key2_spc);

return md5($key1_sec.$key2_sec.$l8b,1);

}


この検証方法については、際限なく文句を言うしかありません! nodeJsのWebSocket操作方法と比較:



コードをコピー

コードは次のとおりです:

//サーバープログラムvar crypto = require('crypto');var WS = '258EAFA5-E914-47DA- 95CA- C5AB0DC85B11';require('net').createServer(function(o){
var key;
o.on('data',function(e){
if(!key){
//handshake
key = e.toString().match(/Sec-WebSocket-Key: (.+)/)[1];
key = crypto.createHash('sha1').update(key + WS).digest('base64 ') ;
o.write('HTTP/1.1 101 スイッチング プロトコルrn');
o.write('アップグレード: websocketrn');
o.write('接続: アップグレード');
o.write('Sec- WebSocket- Accept: ' + key + 'rn');
o.write('rn');
}else{
console.log(e);
};
});
}).listen(8000) ;

2. データフレーム解析コード

この記事では、decodeFrame などのデータ フレーム解析コードは提供しません。データ フレームの形式は、純粋に物理的な作業です。

www.bkjia.comtru​​ehttp://www.bkjia.com/PHPjc/740662.html技術記事以下に、クライアントとサーバー間の WebSocket 接続を確立するときのハンドシェイク部分を示す図を描きました。この部分は、ノードによって提供されるネット モジュールにより、ノード内で非常に簡単に完了できます...
声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。