windows11下拔掉网线,60秒内插上网线,通信恢复【c++ tcp实现】。
前言
当TCP的Peer A ,Peer B 两端建立了连接之后,如果一端突然拔掉网线或拔掉电源时,怎么检测到拔掉网线或者拔掉电源、链路不通?原因是在需要长连接的网络通信程序中,经常需要心跳检测机制,来实现检测对方是否在线或者维持网络连接的需要。
同时如果在网线被短时间内拔掉后又恢复插入,此时通信是否能够恢复到正常的数据收发(不是指重新连接,重建新的socket)。
必要的条件
要能够在短时间内拔插网线后自动的恢复原来的正常通信状态,那么就必须满足下面的铁律:
1)如果网线断开的时间短暂,在SO_KEEPALIVE设定的探测时间间隔内,并且两端在此期间没有任何针对此长连接的网络操作。当连上网线后此TCP连接可以自动恢复,继续进行正常的网络操作。
2)如果网线断开的时间很长,超出了SO_KEEPALIVE设定的探测时间间隔,或者两端期间在此有了任何针对此长连接的网络操作。当连上网线时就会出现ETIMEDOUT或者ECONNRESET的错误。你必须重新建立一个新的长连接进行网络操作。
关键实现片段:
首先我们来看实现客户端的关键函数或内容:
// 监测以及数据接收线程
DWORD WINAPI monitorThread(LPVOID pM)
{
while(1)
{
char szRecvBuf[10] = {
0};
int nRet = recv(sockClient, szRecvBuf, 1, MSG_PEEK); // 注意, 最后一个参数必须是MSG_PEEK, 否则会影响主线程接收信息,这里其实是预读,获取接收数据的缓冲区中是否存在有效数据,因为这里用来判别数据是否存在,因此只需读取1字节
if(nRet <= 0) // 实际上, 等于0表示服务端主动关闭通信socket
{
//判断关闭或者异常的代码区
if(nRet==0) {
std::cout << "server exec close client socket" << std::endl;
}else if(nRet==-1) {
std::cout << "other error nRet is " << nRet << std::endl;
}
closesocket(sockClient);
break;
}
else {
//实际的接收数据的地方
char buf[1024] = {
0};
int readSize = recv(sockClient, buf, 1024, 0);
std::cout << "recv bytes " << readSize << " , data:" << buf << std::endl;
}
Sleep(200);
}
return 0;
}
为了模拟实际的通信交互,开启一个模拟发送的线程
//该线程模拟数据的发送,以便于测试真实的通信发送
DWORD WINAPI monitorSendThread(LPVOID pM)
{
while(1) {
int s = sockClient ;
auto curTime = system_clock::now();
double xx = std::chrono::duration<double,std::milli>
(curTime-sendPoint).count();
if (xx>=60*1000) {
//这里是为了模拟间隙性的发送
char buf[100] = {
0};
memcpy(buf, "1234567890", strlen("1234567890"));
if

最低0.47元/天 解锁文章
2376






