webscoket绑定php uid,think-凯发k8官方网
websocket 客户端消息的解析
前面我们演示了当客户端连接服务端,会触发连接事件,事件中我们要求返回当前客户端的 fd。当客户端发送消息给服务端,服务端会根据我们的规则将消息发送给指定 fd 的客户端:
app/listener/wsconnect.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
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
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
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
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年技术见证,附赠技术全景图总结
以上是凯发k8官方网为你收集整理的webscoket绑定php uid,think-swoole之websocket客户端消息解析与使用socketio处理用户uid与fd关联...的全部内容,希望文章能够帮你解决所遇到的问题。
- 上一篇:
- 下一篇: