内容简介:最后我们再处理一下用户断开连接的时候的处理工作。首先,我们需要在websocket断开的时候,去调用MessageController中的close函数,传递当前用户的fd.在MessageController中,去调用处理用户登录注册所有逻辑的LoginService处理具体的业务逻辑,删除用户和直播的关系。其中包括删除user_id 对应的fd,group对应的fd等的内容。
最后我们再处理一下用户断开连接的时候的处理工作。
首先,我们需要在websocket断开的时候,去调用MessageController中的close函数,传递当前用户的fd.
$this->server->on('close', function ($ser, $fd) { echo "server: close a fd : fd{$fd}\n"; $this->MessageController->closeFd($fd); });
在MessageController中,去调用处理用户登录注册所有逻辑的LoginService处理具体的业务逻辑,删除用户和直播的关系。其中包括删除user_id 对应的fd,group对应的fd等的内容。
//关闭连接 public function closeFd($fd) { $loginService = new LoginService($this->server); $loginService->close($fd); }
然后我们去LoginService中完成逻辑处理:
public function close($fd) { try{ $result=$this->removeFromRedis($fd); if(!$result){ throw new \Exception('删除用户失败,请重试!'); } //在这里也加上notify,如果有需要通知的内容,可以在前面直接写,不用再更改此处的业务逻辑 $this->notify(); }catch (\Exception $e){ $this->result['status'] = 1; $this->result['msg'] = $e->getMessage(); echo json_encode($this->result,JSON_UNESCAPED_UNICODE); } return $this->result; }
具体的实现方法如下:
private function removeFromRedis($fd){ $user_id = pool::redis()->hGet(self::user_bind_redis_key, $fd); pool::redis()->hDel(self::user_bind_redis_key, $fd); //删除当前用户所对应的分组 $rk = str_replace('fd', $fd, self::redis_key_user_group); $group = pool::redis()->get($rk); pool::redis()->del($rk); //删除分组中对应的用户 if ($group) { $group = explode("_", $group); //从分组对应的用户中,删除当前用户 $rk = str_replace(['first_topic', 'second_topic'], [$group[0], $group[1]], self::redis_key_group_user); $userlist = pool::redis()->get($rk); $userlist = explode(',', $userlist); $key = array_search($user_id, $userlist); array_splice($userlist, $key, 1); if (!$userlist) { pool::redis()->del($rk); } else { pool::redis()->set($rk, implode(",", $userlist)); } } //删除fd绑定的用户。 pool::redis()->hDel(self::fd_bind_user_redis_key, $user_id); return true; }
在这段逻辑中主要是通过fd来查找之前我们生成的所有缓存内容,将所有与当前fd相关的内容全部删除。也就使得用户与我们的业务脱离了关系。
当两个用户都连接之后,我们可以看到,redis中一共有如下几个Key:
1) "ws_topic_1_1" 2) "ws_user_2" 3) "ws_fd_bind_user_redis_key" 4) "ws_user_bind_fd_redis_key" 5) "ws_user_4"
我们打开其中的一个去查看,可以发现,两个用户ID正好对应两个fd
127.0.0.1:6379> hkeys ws_fd_bind_user_redis_key 1) "10086" 2) "10087" 127.0.0.1:6379> hget ws_fd_bind_user_redis_key 10086 "2" 127.0.0.1:6379> hget ws_fd_bind_user_redis_key 10087 "4"
然后我们试着关闭10087的用户
再去查看 redis 中的内容
127.0.0.1:6379> keys * 1) "ws_topic_1_1" 2) "ws_user_2" 3) "ws_fd_bind_user_redis_key" 4) "ws_user_bind_fd_redis_key" 127.0.0.1:6379> hkeys ws_fd_bind_user_redis_key 1) "10086" 127.0.0.1:6379> hkeys ws_user_bind_fd_redis_key 1) "2" 127.0.0.1:6379>
所有关于用户10087的内容都已经被删除了。
至此,整个业务流程全部完成。
项目已经放在了github上,欢迎大家批评指正。 https://github.com/gsbhx/swoole_chat
下一步,计划使用Golang实现一下这一功能。
以上所述就是小编给大家介绍的《SWOOLE开发实时聊天系统(十)用户断开聊天》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!
猜你喜欢:- PostgreSQL复制断开故障
- librdkafka 打印连接断开错误
- 俄罗斯成功断开互联网
- 如何使用 RST 来解决 TCP 断开问题
- c# – 以编程方式连接和断开USB设备
- 解决Mac下SSH闲时自动断开的问题
本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
SHA 加密
SHA 加密工具
HEX HSV 转换工具
HEX HSV 互换工具