QT项目复习5——心跳包以及多线程加锁

四、客户端

  • 现在的程序还不够健全,比如——
  • 服务器正常断开后,客户端还是正常的状态

所以需要一个断开的链接信号——

  • 服务器断开会有一个信号
  • 服务器网络异常(网络断开)的情况——

但是网断开后,已经接收不到数据了,但是服务器并没有退出客户端也没有任何改变

  • 解决方法:定时,如果某一段时间之内没收到数据,就关掉客户端
    设置一个心跳包(服务器和客户端隔一段时间就通信一次,证明还存活着),如果心跳包好几秒没收到,那么server对应的套接字就可以关闭了。

因为现在已经有了一个定时器,所以我们可以定义一个变量来表示多少次没有发送心跳包了,再定义一个(5次没有心跳包就关闭)

  • 定时器对应的是更新在线用户列表,所以我们可以将心跳包在线用户列表相关联(每发一次用户列表就加1)——

心跳包重置为0的时机——

当收到服务器心跳包的应答的时候——

四、服务器

  • 服务器的心跳包如何设置,从而在断网后可以关闭服务器
  • 难点之一——
    服务器如何定时呢?
  • alarm()只能一个时刻产生一个定时器,用一个定时器不能处理多个客户端的网络状况的。

所以最好的一个客户端对应一个定时器,这样当产生了问题的时候,就可以把套接字关掉了。

  • 所以这里使用多路IO复用的超时

一个客户端对应一个fd,对应一个反应堆,从而对应一个超时时间

超时的地方一般是阻塞在read()那里,因为没有数据读,所以可以用poll对read()进行封装。

poll作为单个套接字的反应堆——


如果超时了,就让ret为0,从而可以让服务器关闭套接字

  • 服务器加锁
    大部分客户端都在读(获取在线用户列表),只有登录(添加)和删除(退出 )的时候才是写——
    读写锁定义成全局变量
  • 删除的时候要加锁,这个时候是写锁定——
  • 插入的时候也要加锁,登录的时候也是写锁定——
  • 读取链表的时候也要加锁,不过是读锁定——
  • 考虑一下管道破裂的情况——
  • 第一次写是不会破裂的,只有第二次写的时候会破裂

这里只写一次数据write,就算正在写的时候读端关闭了,还是照常写,只不过是收到一个RST

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 二、客户端 登录功能的话是发送登录数据包sendLogin(),那么获取用户列表也要发送列表数据包sendList...
    StevenHD阅读 4,666评论 0 0
  • 最近在看socket编程的一些内容,想总结下。先从理论知识总结下,不涉及代码。 1.socket是什么? sock...
    令__狐冲阅读 8,193评论 1 5
  • UDP:用户数据报协议:主要用在实时性要求比较高的以及对质量相对较弱的地方.但是面对现在高质量的线路不会容易丢包,...
    CoderZS阅读 12,074评论 0 24
  • 心跳机制是定时发送一个自定义的结构体(心跳包),让对方知道自己还活着,以确保连接的有效性的机制。 网络中的接收...
    踩在浪花上00阅读 4,183评论 0 0
  • 什么是心跳机制? 心跳说的是在客户端和服务端在互相建立ESTABLISH状态的时候,如何通过发送一个最简单的包来保...
    tracy_668阅读 10,452评论 1 5