最新文章专题视频专题问答1问答10问答100问答1000问答2000关键字专题1关键字专题50关键字专题500关键字专题1500TAG最新视频文章推荐1 推荐3 推荐5 推荐7 推荐9 推荐11 推荐13 推荐15 推荐17 推荐19 推荐21 推荐23 推荐25 推荐27 推荐29 推荐31 推荐33 推荐35 推荐37视频文章20视频文章30视频文章40视频文章50视频文章60 视频文章70视频文章80视频文章90视频文章100视频文章120视频文章140 视频2关键字专题关键字专题tag2tag3文章专题文章专题2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章专题3
当前位置: 首页 - 科技 - 知识百科 - 正文

Think-Swoole之WebSocket客户端消息解析与使用SocketIO处理用户UID与fd关联

来源:动视网 责编:小采 时间:2020-11-02 18:01:55
文档

Think-Swoole之WebSocket客户端消息解析与使用SocketIO处理用户UID与fd关联

Think-Swoole之WebSocket客户端消息解析与使用SocketIO处理用户UID与fd关联:WebSocket 客户端消息的解析前面我们演示了当客户端连接服务端,会触发连接事件,事件中我们要求返回当前客户端的 fd。当客户端发送消息给服务端,服务端会根据我们的规则将消息发送给指定 fd 的客户端:app/listener/WsConnect.php<php
推荐度:
导读Think-Swoole之WebSocket客户端消息解析与使用SocketIO处理用户UID与fd关联:WebSocket 客户端消息的解析前面我们演示了当客户端连接服务端,会触发连接事件,事件中我们要求返回当前客户端的 fd。当客户端发送消息给服务端,服务端会根据我们的规则将消息发送给指定 fd 的客户端:app/listener/WsConnect.php<php

WebSocket 客户端消息的解析

前面我们演示了当客户端连接服务端,会触发连接事件,事件中我们要求返回当前客户端的 fd。当客户端发送消息给服务端,服务端会根据我们的规则将消息发送给指定 fd 的客户端:

app/listener/WsConnect.php

<?php
declare (strict_types = 1);
namespace applistener;
class WsConnect
{
 /**
 * 事件监听处理
 *
 * @return mixed
 * 受用 WebSocket 客户端连接入口
 */
 public function handle($event)
{
 //实例化 Websocket 类
 $ws = app('	hinkswooleWebsocket');
 //
 $ws -> emit('sendfd',$ws -> getSender());
 }
}

app/listener/WsTest.php

<?php
declare (strict_types = 1);
namespace applistener;
use 	hinkswooleWebsocket;
class WsTest
{
 /**
 * 事件监听处理
 *
 * @return mixed
 */
 public function handle($event,Websocket $ws)
{
 $ws -> to(intval($event['to'])) -> emit('testcallback',$event['message']);
 }
}

客户端执行上述两个事件后,控制台打印出以下信息:

3e65bb5742e35710db2e3ae1e9bba30.png

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

因为我们使用的扩展是基于 SocketIO 协议的,这些数字可以理解为协议的代号。

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

ac3e617f79cdf6b208b304f2ca64911.png

上面是 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>

解析后的数据:

d049d27768d703d197019f59ceac116.png

使用 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 applistener;
class WsConnect
{
 /**
 * 事件监听处理
 *
 * @return mixed
 * 受用 WebSocket 客户端连接入口
 */
 public function handle($event)
{
 //实例化 Websocket 类
 $ws = app('	hinkswooleWebsocket');
 //
 $ws -> emit('sendfd',$ws -> getSender());
 }
}

app/listener/WsTest.php

<?php
declare (strict_types = 1);
namespace applistener;
use 	hinkswooleWebsocket;
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:

a15a44c4dc464e480664ac7add1efb2.png

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 发送信息:

73542665974ef733eb1a7afd192a7f5.png

两个客户端均会受到信息:

1752f5ebc05df83efbe7fc07a0c55f8.png

可见消息已经经过解析,因为 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 applistener;
class WsConnect
{
 /**
 * 事件监听处理
 *
 * @return mixed
 * 受用 WebSocket 客户端连接入口
 */
 public function handle($event)
{
 // $event 为请求对象
 //实例化 Websocket 类
 $ws = app('	hinkswooleWebsocket');
 //获取 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客户端消息解析与使用SocketIO处理用户UID与fd关联

Think-Swoole之WebSocket客户端消息解析与使用SocketIO处理用户UID与fd关联:WebSocket 客户端消息的解析前面我们演示了当客户端连接服务端,会触发连接事件,事件中我们要求返回当前客户端的 fd。当客户端发送消息给服务端,服务端会根据我们的规则将消息发送给指定 fd 的客户端:app/listener/WsConnect.php<php
推荐度:
标签: 消息 用户 客户端
  • 热门焦点

最新推荐

猜你喜欢

热门推荐

专题
Top