我正在调试基于c的linux套接字程序.正如网站上提供的所有示例一样,
我应用了以下结构:
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);
当删除服务器关闭其服务器程序时,我可以检测到断开连接.但是,如果我拔下以太网电缆,send函数仍然返回正值而不是-1.
假设我无法更改服务器端,如何检查客户端程序中的网络连接?
解决方法:
But if I unplug the ethernet cable, the send function still return
positive values rather than -1.
首先你应该知道send实际上并没有发送任何东西,它只是一个内存复制功能/系统调用.它将数据从您的进程复制到内核 – 稍后内核将获取该数据,并在将数据打包成段和数据包后将其发送到另一端.因此,在以下情况下,send只能返回错误:
>套接字无效(例如伪造文件描述符)
>连接显然无效,例如它尚未建立或已经以某种方式终止(FIN,RST,超时 – 见下文)
>没有更多的空间来复制数据
重点是send不发送任何内容,因此其返回代码不会告诉您有关实际到达另一方的数据的任何信息.
回到你的问题,当TCP发送数据时,它需要在合理的时间内得到有效的确认.如果没有,则重新发送.它多久重发一次?每个TCP堆栈的工作方式都不同,但规范是使用指数退避.也就是说,首先等待1秒,然后是2,然后是4,依此类推.在某些堆栈上,此过程可能需要几分钟.
重点是,在中断的情况下,TCP将在严重的沉默期后宣布连接死亡(在Linux上它执行15次重试 – 超过5分钟).
解决此问题的一种方法是在您的应用程序中实现一些确认机制.例如,您可以向服务器发送请求“在5秒内回复,或者我将声明此连接已死”,然后以超时方式重新启动.
标签:c-3,send,linux,sockets,disconnection
来源: https://codeday.me/bug/20191004/1851661.html