class Event
{
public static $timer_id;
public static function onMessage($client_id, $message)
{
switch($message_data)
{
case 're_login': //登录时创建定时器
$time_interval =60;
self::$timer_id = \Workerman\Lib\Timer::add($time_interval, function(){
//要执行的逻辑
var_dump(111);
Gateway::sendToClient(...)//给对应的房间对应的客户端发送一条随机消息
}
return;
}
}
public static function onClose($client_id)
{
\Workerman\Lib\Timer::del(self::$timer_id);//客户端退出时清除定时器
}
}
代码貌似错了,当客户端退出时无法清除定时器。控制台到时间照样输出111。大神们帮忙解决一下呗。
1、businessWorker是多进程的,客户端发来的请求默认是gateway随机转发给businessWorker处理,
也就是创建定时器的进程和运行onClose删除定时器的进程不是同一个进程,会导致删除定时器失败。
解决方法:设置gateway到businessWorker的路由,让同一个客户端的请求都转发给某个固定的businessWorker进程,可以直接使用gatewayWorker手册路由
http://workerman.net/gatewaydoc/gateway-worker-development/router.html
中的范例 2 随机绑定来达到这一目的。
2、按照楼主需求,定时器id $timer_id 不能放在类的静态常量中,因为这样不管有多少各客户端设置了定时器,这里的值只有一个,也就是只保存了最后一个定时器的$timer_id ,前面的都被覆盖了。
解决方法:将定时器的$timer_id 存储在$_SESSION中
GatewayWorker $_SESSION参见手册 http://workerman.net/gatewaydoc/gateway-worker-development/session.html
public static function onMessage(..)
{
...
case 're_login': //登录时创建定时器
$_SESSION = \Workerman\Lib\Timer::add(...
...
}
public static function onClose(..)
{
\Workerman\Lib\Timer::del($_SESSION);
}
最后建议带着需求把手册多看几遍再来写程序,可以大大加快开发速度
谢谢walkor