避免TIME_WAIT

我试图在客户端避免TIME_WAIT。 我连接然后设置O_NONBLOCK和SO_REUSEADDR。 我调用read直到它返回0.当read返回0时,errno也是0.我将此解释为服务器关闭连接的标志。 但是,如果我调用close,则套接字设置为TIME_WAIT,由netstat确认。

由于我与同一主机/端口建立了多个连接,因此我最终开始看到“使用中的地址”错误(请参阅http://hea-www.harvard.edu/~fine/Tech/addrinuse.html )。

读取返回0后,我应该关闭吗? 如果不这样,文件描述符会被释放吗?

启动关闭连接的那一方是最终处于TIME_WAIT状态的一方。 read()返回0应该指示服务器首先关闭套接字,所以是 – 这应该意味着TIME_WAIT最终在服务器端,并且客户端通过LAST_ACK

在一天结束时,您无法避免TIME_WAIT状态。 即使您成功将其从客户端移动到服务器端,在TIME_WAIT结束之前,您仍然无法重新使用它(server host, server port, client host, client port)元组(无论它在哪一侧) )。

由于在您的场景( server hostserver portclient host )中修复了该元组的三个部分,因此您实际上只有以下选项:

  • 尝试提供更多客户端端口。 默认情况下,某些操作系统仅为“临时端口”使用一小部分可用端口(在这方面我不确定OSX)。 如果是这种情况,请查看是否可以通过操作系统中的配置调整来更改范围,或者让应用程序在循环中搜索带有bind() / connect()的工作端口,直到连接正常工作。

  • 通过在客户端上使用多个IP地址,扩展可用的client host值的数量。 您必须具有应用程序bind() ,特别是其中一个IP地址。

  • 通过在server host使用多个端口和/或IP地址,扩展可用的server host / server port值的数量。 客户端需要选择一个连接(循环,随机等)。

  • 可能是最好的选择,如果它是可行的:重构你的协议,以便完成的连接不会关闭,但进入“空闲”状态,以便以后可以重用它们,而不是打开一个新的连接(如HTTP)活着)。

稍后在同一页面上他们提到了SO_REUSEADDR。 这就是你所需要的。 您肯定希望在返回零时关闭读取文件描述符。

在客户端设置SO_REUSEADDR对服务器端没有帮助,除非它还设置了SO_REUSEADDR