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

5a7f49afa4d35b2e85dd4741ec277c26.png

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']);

}

}

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

15db0b21485cd5a649ed1c425c9efd6b.png

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

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

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

2c4b4121843663a7cfc5604eb01b4119.png

上面是 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',数据] 这种格式

}

解析后的数据:

e800f4bd71d56e28f6637851691241cd.png

使用 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:

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

7cfb2a6898ec600ce334259965fd25db.png

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

faeaf12d279e866016f20e6082786d14.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.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 来映射二者关系也是不错的选择。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
think-swoole 是一个基于 Swoole 扩展的 PHP 框架,它提供了对 WebSocket 的支持。下面是一个简单的示例: ```php // 定义 WebSocket 控制器类 namespace app\index\controller; use think\swoole\WebSocket; class Chat extends WebSocket { protected $server; // WebSocket 连接建立时触发的事件 public function onOpen($server, $request) { $this->server = $server; $this->send($request->fd, 'Welcome to Chat Room!'); } // WebSocket 接收到消息时触发的事件 public function onMessage($server, $frame) { $data = $frame->data; foreach ($server->connections as $fd) { $this->send($fd, $data); } } // WebSocket 连接关闭时触发的事件 public function onClose($server, $fd) { echo "client-{$fd} is closed\n"; } } ``` 在上面的代码中,我们定义了一个名为 Chat 的 WebSocket 控制器类,它继承了 think\swoole\WebSocket 类,并实现了三个事件方法 onOpen、onMessage 和 onClose。在 onOpen 方法中,我们向客户端发送了一条欢迎信息;在 onMessage 方法中,我们将接收到的消息发送给所有连接的客户端;在 onClose 方法中,我们输出了客户端关闭的信息。 接下来,在路由配置文件中定义 WebSocket 路由: ```php // 定义 WebSocket 路由 use think\facade\Route; Route::get('/ws', 'index/Chat'); ``` 最后,在 Swoole 服务器配置文件中开启 WebSocket 服务: ```php // Swoole 服务器配置文件 use think\swoole\Server; return [ 'host' => '0.0.0.0', 'port' => 9501, 'type' => Server::SERVER_TYPE_WEB_SOCKET, 'mode' => SWOOLE_PROCESS, 'sock_type' => SWOOLE_SOCK_TCP, 'option' => [ 'worker_num' => 8, 'daemonize' => false, 'backlog' => 128, ], ]; ``` 在上面的代码中,我们将服务器的类型设置为 WebSocket,并开启了 8 个 Worker 进程。 现在,我们就可以启动 Swoole 服务器,并在客户端连接到 WebSocket 服务后进行消息的发送和接收了。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值