>[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(); ```