日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > php >内容正文

php

webscoket绑定php uid,Think-Swoole之WebSocket客户端消息解析与使用SocketIO处理用户UID与fd关联...

發布時間:2024/10/12 php 56 豆豆
生活随笔 收集整理的這篇文章主要介紹了 webscoket绑定php uid,Think-Swoole之WebSocket客户端消息解析与使用SocketIO处理用户UID与fd关联... 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

WebSocket 客戶端消息的解析

前面我們演示了當客戶端連接服務端,會觸發連接事件,事件中我們要求返回當前客戶端的 fd。當客戶端發送消息給服務端,服務端會根據我們的規則將消息發送給指定 fd 的客戶端:

app/listener/WsConnect.php<?php

declare (strict_types = 1);

namespace app\listener;

class WsConnect

{

/**

* 事件監聽處理

*

* @return mixed

* 受用 WebSocket 客戶端連接入口

*/

public function handle($event)

{

//實例化 Websocket 類

$ws = app('\think\swoole\Websocket');

//

$ws -> emit('sendfd',$ws -> getSender());

}

}

app/listener/WsTest.php<?php

declare (strict_types = 1);

namespace app\listener;

use \think\swoole\Websocket;

class WsTest

{

/**

* 事件監聽處理

*

* @return mixed

*/

public function handle($event,Websocket $ws)

{

$ws -> to(intval($event['to'])) -> emit('testcallback',$event['message']);

}

}

客戶端執行上述兩個事件后,控制臺打印出以下信息:

返回信息前面有一些數字,40、42都代表什么意義呢?

因為我們使用的擴展是基于 SocketIO 協議的,這些數字可以理解為協議的代號。

打開 /vendor/topthink/think-swoole/src/websocket/socketio/Packet.php ,有以下內容:

上面是 Socket 類型,下面是引擎,前后兩個代號上下拼湊得到:40:”MESSAGE CONNECT”

42:”MESSAGE EVENT”

結合這些代碼,能知道 SocketIO 中消息的大體運作情況。

通過控制臺打印出的消息,我們發現這些消息不能直接拿到使用,需要進行截取處理:

test.html

Document

消息:

接收者:

發送

var ws = new WebSocket("ws://127.0.0.1:9501/");

ws.onopen = function(){

console.log('連接成功');

}

//數據返回的解析

function mycallback(data){

var start = data.indexOf('[') // 第一次出現的位置

var start1 = data.indexOf('{')

if(start < 0){

start = start1;

}

if(start >= 0 && start1 >= 0){

start = Math.min(start,start1);

}

if(start >= 0){

console.log(data);

var json = data.substr(start); //截取

var json = JSON.parse(json);

console.log(json);

}

}

ws.onmessage = function(data){

// console.log(data.data);

mycallback(data.data);

}

ws.onclose = function(){

console.log('連接斷開');

}

function send()

{

var message = document.getElementById('message').value;

var to = document.getElementById('to').value;

console.log("準備給" + to + "發送數據:" + message);

ws.send(JSON.stringify(['test',{

to:to,

message:message

}])); //發送的數據必須是 ['test',數據] 這種格式

}

解析后的數據:

使用 SocketIO 處理消息業務

SocketIO 的相關知識可以查看文檔,重點看客戶端方面知識:

https://www.w3cschool.cn/socket/socket-k49j2eia.html

iotest.html

Document

消息:

接收者:

發送

//http 協議

var socket = io("http://127.0.0.1:9501", {transports: ['websocket']});

socket.on('connect', function(){

console.log('connect success');

});

socket.on('close',function(){

console.log('connect close')

});

//send_fd 為自定義的場景值,和后端對應

socket.on("sendfd", function (data) {

console.log(data)

});

//testcallback 為自定義的場景值,和后端對應

socket.on("testcallback", function (data) {

console.log(data)

});

function send() {

var message = document.getElementById('message').value;

var to = document.getElementById('to').value;

socket.emit('test', {

//屬性可自行添加

to:to,

message:message

})

}

var socket = io("http://127.0.0.1:9501", {transports: ['websocket']}); 中第二個參數指明要升級的協議。

app/listener/WsConnect.php<?php

declare (strict_types = 1);

namespace app\listener;

class WsConnect

{

/**

* 事件監聽處理

*

* @return mixed

* 受用 WebSocket 客戶端連接入口

*/

public function handle($event)

{

//實例化 Websocket 類

$ws = app('\think\swoole\Websocket');

//

$ws -> emit('sendfd',$ws -> getSender());

}

}

app/listener/WsTest.php<?php

declare (strict_types = 1);

namespace app\listener;

use \think\swoole\Websocket;

class WsTest

{

/**

* 事件監聽處理

*

* @return mixed

*/

public function handle($event,Websocket $ws)

{

// $ws -> to(intval($event['to'])) -> emit('testcallback',$event['message']);

$ws -> to(intval($event['to'])) -> emit('testcallback',[

'form' => [

'id' => 10,

'fd' => $ws -> getSender(),

'nickname' => '張三'

],

'to' => [

'id' => 11,

'fd' => intval($event['to']),

'nickname' => '李四'

],

'massage' => [

'id' => 888,

'create_time' => '2020-03-13',

'content' => $event['message']

]

]);

}

}

開啟兩個客戶端,fd 分別是5、6:

WsConnect.php 中,有 $ws -> emit('sendfd',$ws -> getSender()); 發送 fd 消息對應的場景值是 “sendfd” ,在 iotest.html 中,有socket.on("sendfd", function (data) {console.log(data)}); 這段代碼,其中也有場景值 “sendfd”,這行代碼可以直接獲取對應場景值的信息,所以控制臺上會打印出 fd 值。

用 fd 5 向 fd 6 發送信息:

兩個客戶端均會受到信息:

可見消息已經經過解析,因為 WsTest.php 中 發送消息指定場景值 testcallback,iotest.html 中通過 socket.on("testcallback", function (data){console.log(data)}); 可直接獲取解析過的結果。

這就看出了 SocketIO 在客戶端消息接收方面的便捷之處了。

用戶 UID 和客戶端 fd 的綁定

前面的例子中,都是通過指定 fd 來向客戶端發送消息,實際場景中,我們不可能通過 fd 確定發送對象,因為 fd 不是固定不變的,因此需要將用戶的 UID 與客戶端的 fd 進行綁定,進而可以通過選擇用戶,來確定 fd 完成消息的發送。

只需要將前端頁面的 HTTP 連接中增加 UID 參數即可:

test.htmlvar ws = new WebSocket("ws://127.0.0.1:9501/?uid=1");

iotest.htmlvar socket = io("http://127.0.0.1:9501?uid=1", {transports: ['websocket']});

后端可以在連接事件中進行綁定:

app/listener/WsConnect.php<?php

declare (strict_types = 1);

namespace app\listener;

class WsConnect

{

/**

* 事件監聽處理

*

* @return mixed

* 受用 WebSocket 客戶端連接入口

*/

public function handle($event)

{

// $event 為請求對象

//實例化 Websocket 類

$ws = app('\think\swoole\Websocket');

//獲取 uid

$uid = $event -> get('uid');

//獲取 fd

$fd = $ws -> getSender();

//獲取到 uid 和 fd 后,可以存數據庫,內存或者 redis

$ws -> emit('sendfd',[

'uid' => $uid,

'fd' => $fd

]);

}

}

有了 UID 與 fd ,可以在每次連接成功后,更新數據庫,連接斷開后再清空用戶對因的 fd。假如服務器重啟,那么二者的對應關系也就沒用了,所以不必存入數據庫,存入 Redis 最好,通過 Redis 的 Hash 來映射二者關系也是不錯的選擇。

與50位技術專家面對面20年技術見證,附贈技術全景圖

總結

以上是生活随笔為你收集整理的webscoket绑定php uid,Think-Swoole之WebSocket客户端消息解析与使用SocketIO处理用户UID与fd关联...的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。