Linux Socket:如何在客户端程序中检测断开的网络?

use*_*197 13 c sockets linux send disconnection

我正在调试基于ac的linux socket程序.正如网站上提供的所有示例一样,我应用了以下结构:

sockfd= socket(AF_INET, SOCK_STREAM, 0);

connect(sockfd, (struct sockaddr *) &serv_addr, sizeof(serv_addr));

send_bytes = send(sockfd, sock_buff, (size_t)buff_bytes, MSG_DONTWAIT);
Run Code Online (Sandbox Code Playgroud)

当删除服务器关闭其服务器程序时,我可以检测到断开连接.但是如果我拔掉以太网电缆,send函数仍会返回正值而不是-1.

假设我无法更改服务器端,如何检查客户端程序中的网络连接?

cni*_*tar 37

但是如果我拔掉以太网电缆,send函数仍会返回正值而不是-1.

首先,您应该知道send实际上并没有发送任何内容,它只是一个内存复制功能/系统调用.它从过程到内核拷贝数据-晚些时候内核将获取数据并在段和包包装之后发送给对方.因此send,只有在以下情况下才能返回错

  • 套接字无效(例如伪造文件描述符)
  • 连接显然无效,例如它尚未建立或已经以某种方式终止(FIN,RST,超时 - 见下文)
  • 没有更多的空间来复制数据

重点是send不发送任何内容,因此其返回代码不会告诉您有关实际到达另一方的数据的任何信息.

回到你的问题,当TCP发送数据时,它需要在合理的时间内得到有效的确认.如果没有,则重新发送.它多久重发一次?每个TCP堆栈的工作方式都不同,但规范是使用指数退避.也就是说,首先等待1秒,然后是2,然后是4,依此类推.在某些堆栈上,此过程可能需要几分钟.

重点是,在中断的情况下,TCP将在严重的沉默期后宣布连接死亡(在Linux上它执行15次重试 - 超过5分钟).

解决此问题的一种方法是在您的应用程序中实现一些确认机制.例如,您可以向服务器发送请求" 在5秒内回复,或者我将声明此连接已死 ",然后recv超时.