Qt使用Keepalive机制与断线重连 |
您所在的位置:网站首页 › tcp断线重连机制 › Qt使用Keepalive机制与断线重连 |
QTcpSocket能够检测到Socket的连接与断开状态,并触发相关信号,我们只需要关联信号与槽就能够知道连接状态。 还有一些特殊情况是无法触发QTcpSocket::disconnected信号,比如说:网线突然拔了、对端设备突然爆掉了等。这类情况由于对端socket未正常调用close()方法而导致的。 我们可以定义一个心跳包去定期检查对端的存活状态,这种做法在协议还未指定的初期是比较适合的,客户端与服务端制定一套心跳请求与应答机制来判断对端的存活状态。但是往往下位机的程序已经存在(开发者不愿意修改或增加现有协议),这时候如何能够在特殊情况下检测到网络断开呢? 主角登场:Keepalive机制 keepalive简介(摘自维基百科)传输控制协议(TCP)存活包为可选特性,且默认关闭。存活包内没有数据。在以太网网络中,存活包的大小为最小长度的几帧(64字节)。协议中,还有三个与存活包相关的参数: 存活时长(英语:Keepalive time)即空闲时,两次传输存活包的持续时间。TCP存活包时长可手动配置,默认不少于2个小时。 存活间隔(英语:Keepalive interval)即未收到上个存活包时,两次连续传输存活包的时间间隔。 存活重试次数(英语:Keepalive retry)即在判断远程主机不可用前的发送存活包次数。当两个主机透过TCP/IP协议相连时,TCP存活包可用于判断连接是否可用,并按需中断。多数支持TCP协议的主机也同时支持TCP存活包。每个主机按一定周期向其他主机发送TCP包来请求回应。若发送主机未收到特定主机的回应(ACK),则将从发送主机一侧中断连接。 若其他主机在连接关闭后发送TCP存活包,关闭连接的一方将发送RST包来表明旧连接已不可用。其他主机将关闭它一侧的连接以新建连接。 空闲的TCP连接通常会隔每45秒或60秒发送一次存活包。在未连续收到三次ACK包时,连接将中断。此行为因主机而异,如默认情况下的Windows主机将在7200000ms(2小时)后发送首个存活包,随后再以1000ms的间隔发送5个存活包。若任意存活包未收到回应,连接将被中断。 Qt开启Keepalive(Linux与Windows)Socket文件描述符的获取QAbstractSocket::socketDescriptor(),在socket连接成功后可使通过m_socket->socketDescriptor();获取到QTcpSocket的文件描述符(FD),失败时返回-1,这边获取到的fd可以提供给int enableKeepalive(int fd);作为参数用于启用keepalive。 文档介绍: Returns the native socket descriptor of the QAbstractSocket object if this is available; otherwise returns -1. If the socket is using QNetworkProxy, the returned descriptor may not be usable with native socket functions. The socket descriptor is not available when QAbstractSocket is in UnconnectedState. 断线重连在开启keepalive后我们就可以在链路断开时触发QTcpSocket::disconnected信号了,这时候我们只需要开启一个定时器去试试检查QTcpSocket的状态,当状态为QAbstractSocket::UnconnectedState时清理QTcpSocket资源并尝试重新与服务端建立连接即可。 |
CopyRight 2018-2019 办公设备维修网 版权所有 豫ICP备15022753号-3 |