>[info] 后端部分代码可以进行用Workerman自带的定时器去处理,自动断链已下线的长连接
关键代码:
```
Timer::add(CHECK_HEARTBEAT_TIME, function()use($worker){
$time_now = time();
global $worker;
foreach($worker->uidConnections as $connection){
// 有可能该connection还没收到过消息,则lastMessageTime设置为当前时间
if (empty($connection->lastMessageTime)) {
$connection->lastMessageTime = $time_now;
continue;
}
// 上次通讯时间间隔大于心跳间隔,则认为客户端已经下线,关闭连接
if ($time_now - $connection->lastMessageTime > HEARTBEAT_TIME) {
$connection->close();
}
}
});
```
全部代码: 此代码是在项目的根目录下面的,里面的text协议是用于内部发送消息用的,代码中有注释
```
<?php
/**
* Created by 老吴.
* UserMsg:砥砺前行,扬帆起航
* email:cwwx0128@qq.com
* QQ:1113249273
* QQ群:925283872
* 微信:cww0128
* Date: 2020/11/19
* Time: 11:14
*/
use Workerman\Worker;
use Workerman\Lib\Timer;
use app\push\controller\Test;
require_once './vendor/workerman/workerman/Autoloader.php';
define('HEARTBEAT_TIME', 30); // 定义一个心跳间隔30秒
define('CHECK_HEARTBEAT_TIME', 1); // 检查连接的间隔时间
// 初始化一个worker容器,监听1234端口
$worker = new Worker('websocket://0.0.0.0:1234');//
/*
* 注意这里进程数必须设置为1,否则会报端口占用错误
* (php 7可以设置进程数大于1,前提是$inner_text_worker->reusePort=true)
*/
$worker->count = 1;
// worker进程启动后创建一个text Worker以便打开一个内部通讯端口
$worker->onWorkerStart = function($worker)
{
//定时器心跳检测下线用户关闭连接
Timer::add(CHECK_HEARTBEAT_TIME, function()use($worker){
$time_now = time();
global $worker;
foreach($worker->uidConnections as $connection){
// 有可能该connection还没收到过消息,则lastMessageTime设置为当前时间
if (empty($connection->lastMessageTime)) {
$connection->lastMessageTime = $time_now;
continue;
}
// 上次通讯时间间隔大于心跳间隔,则认为客户端已经下线,关闭连接
if ($time_now - $connection->lastMessageTime > HEARTBEAT_TIME) {
$connection->close();
}
}
});
// 开启一个内部端口,方便内部系统推送数据,Text协议格式 文本+换行符
$inner_text_worker = new Worker('text://0.0.0.0:5678');
$inner_text_worker->onMessage = function($connection, $buffer)
{
// $data数组格式,里面有uid,表示向那个uid的页面推送数据
$data = json_decode($buffer, true);
$uid = $data['uid'];
// 通过workerman,向uid的页面推送数据
$ret = sendMessageByUid($uid, $buffer);
$connection->send($ret ? 'ok' : 'fail');
};
// $connection->send('你好,你连接我了');
// ## 执行监听 ##
$inner_text_worker->listen();
};
// 新增加一个属性,用来保存uid到connection的映射
$worker->onConnect=function ($connection){
$connection->send('HeartBeat');
};
$worker->uidConnections = array();
// 当有客户端发来消息时执行的回调函数
$worker->onMessage = function($connection, $data)
{
global $worker;
$connection->lastMessageTime = time();
$data=json_decode($data,true);
// 判断当前客户端是否已经验证,既是否设置了uid
if(!isset($connection->uid))
{
// 没验证的话把第一个包当做uid(这里为了方便演示,没做真正的验证)
// $connection->uid = $connection->id;
$connection->uid = $data['id'];
/* 保存uid到connection的映射,这样可以方便的通过uid查找connection,
* 实现针对特定uid推送数据
*/
$worker->uidConnections[$connection->uid] = $connection;
$connection->send('Link_successful');
// sendMessageByUid($data['id'], $data['msg']);
return;
}else{
sendMessageByUid($data['id'], $data['msg']);
$connection->send('Already_Linked');
}
};
// 当有客户端连接断开时
$worker->onClose = function($connection)
{
global $worker;
if(isset($connection->uid))
{
// 通过workerman,向uid的页面推送数据
sendMessageByUid($connection->uid, 123);
// 连接断开时删除映射
unset($worker->uidConnections[$connection->uid]);
}
};
// 向所有验证的用户推送数据
function broadcast($message)
{
global $worker;
foreach($worker->uidConnections as $connection)
{
$connection->send($message);
}
}
// 针对uid推送数据
function sendMessageByUid($uid, $message)
{
global $worker;
if(isset($worker->uidConnections[$uid]))
{
$connection = $worker->uidConnections[$uid];
$connection->send($message);
return true;
}else{
return false;
}
}
// 运行所有的worker
Worker::runAll();
```