這篇文章將為大家詳細講解有關Think-Swoole之WebSocket客戶端消息解析示例,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。
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
<!DOCTYPE HTML>
<html>
<head>
<meta charset="UTF-8">
<title>Document</title>
</head>
<body>
消息:<input type="text" id="message">
接收者:<input type="text" id="to">
<button onclick="send()">發送</button>
<script>
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',數據] 這種格式
}
</script>
</body>
</html>解析后的數據:

使用 SocketIO 處理消息業務
SocketIO 的相關知識可以查看文檔,重點看客戶端方面知識:
https://www.w3cschool.cn/socket/socket-k49j2eia.html
iotest.html
<!DOCTYPE HTML>
<html>
<head>
<meta charset="UTF-8">
<title>Document</title>
</head>
<body>
消息:<input type="text" id="message">
接收者:<input type="text" id="to">
<button onclick="send()">發送</button>
<script src="./socketio.js"></script>
<script>
//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
})
}
</script>
</body>
</html>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.html
var ws = new WebSocket("ws://127.0.0.1:9501/?uid=1");iotest.html
var 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 來映射二者關系也是不錯的選擇。
關于“Think-Swoole之WebSocket客戶端消息解析示例”這篇文章就分享到這里了,希望以上內容可以對大家有一定的幫助,使各位可以學到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。