Closed nameldk closed 7 years ago
session的存取是异步的,所以有一点点延迟(大概小于1毫秒左右)。如果请求过快,可能会造成A请求改变了session,但是B请求读到的仍然是旧的session。为了解决这个问题,可以用Gateway::getSession 和 Gateway::setSession 接口来存取session。
但是onClose里无法调用Gateway::getSession 和 Gateway::setSession接口,因为此时链接已经关闭了。 我想这里的解决办法是在内存中添加一个全局数组,里面记录所有client_id的session快照,如果快照本存在,就用最新的快照。
class Events
{
// ['client_id1'=>session_array, 'client_id2'=>session_array, ...]
public static $sessions = array();
public static function onMessage($client_id)
{
// 如果内存中有对应client_id的session快照数据则使用内存中最新的数据
$_SESSION = isset(self::$sessions[$client_id]) ? self::$sessions[$client_id] : $_SESSION;
// 各种操作$_SESSION
$_SESSION['xxx'] = xx;
unset($_SESSION['rooms']['A']);
// 操作完session记得同步一份给self::$sessions作为快照,保证self::$sessions里的数据是最新的
self::$sessions[$client_id] = $_SESSION;
}
public function onClose($client_id)
{
// 如果内存中有对应client_id的session数据则使用内存中最新的数据
$_SESSION = isset(self::$sessions[$client_id]) ? self::$sessions[$client_id] : $_SESSION;
// 业务逻辑
// ....
// 链接断开了,最后记得删除快照,避免内存泄漏
unset(self::$sessions[$client_id]);
}
}
注意:这个快照要求gateway路由规则是将某个client_id的请求路由给固定worker进程(默认路由就是这样)。 如果你自定义了路由规则需要小心,某个client_id的session数据可能分散在各个worker进程的self::$sessions里,那么这个方法就不会奏效。
了解了,感谢!
1、对 2、性能上一般感觉不到差别 3、对
是在直播间进出房间的场景,用户进入直播间A,我在当前用户的session里记录A $_SESSION['rooms']['A']=1,用户退出A时,我把A从session里unset掉 unset($_SESSION['rooms']['A'])。现在是网页里用过websocket建立连接,用户关闭页面时用户退出A,随即网页关闭,websocket连接断开了。 问题:我会在后端socket断开连接的回调里检测用户是否还有未关闭的房间,有的话会把用户退出房间,可能是 退出房间和socket连接间隔太短了,退出房间时session更新了,但回调时这时session里的A并没清除,从而导致用户会再退出A一次。 Workerman version:3.3.2 GatewayWorker master分支 PHP version:5.6.2 CentOS 7