情形 | 对端进程崩溃 | 对端主机崩溃 | 对端主机不可达 |
1.本端TCP正主动发送数据 | 对端TCP发送一个FIN,这通过使用select判断可读条件立即能检测出来。如果本端TCP继续发送,对端TCP就以RST响应。如果在本端TCP收到RST之后应用进程仍试图写套接口,我们的套接口实现就给该进程发送一个SIGPIPE信号 | 本端TCP将超时,且套接口的待处理错误被设置为ETIMEDOUT | 本端TCP将超时,且接口的待处理错误被设置为EHOSTUNREACH |
2.本端TCP正主动接收数据 | 对端TCP将发送一个FIN,我们将把它作为一个(可能过早的)EOF读入 | 我们将停止接受数据 | 我们将停止收取数据 |
3.连接空闲,保持存活选项(SO_KEEPALIVE)已设 | 对端TCP发送一个FIN,这通过使用select判断可读条件立即能检测出来 | 在毫无动静2小时以后,发送9个保持存活探测包,然后套接口的待处理错误被设置为ETIMEDOUT | 在毫无动静2小时后,发送9个保持存活探测包,然后套接口的待处理错误被设置为EHOSTUNREACH |
4.连接空闲,保持存活选项未设 | 对端TCP发送一个FIN,这通过使用select判断可读条件立即能检测出来 | 无 | 无 |
其中发生对端主机不可达的情况的原因或者是网络故障,或者是对端主机已经崩溃,而最后一跳的路由器也检测到它的崩溃。由路由器向源主机发送ICMP报文。
在面对第3,4种情况时,我们应该在应用程序层对socket设置一个超时机制,避免TCP的半开连接。