尝试进行快速的Socket创建和销毁以进行负载测试时出现“ java.net.BindException:地址已在使用中”


问题内容

我正在尝试通过打开与服务器的大量套接字连接,进行身份验证,关闭连接,然后重复进行负载测试Java服务器。我的应用运行了一段时间,但最终我得到了:

java.net.BindException:地址已在使用中:connect

根据我阅读的文档,其原因是,在调用close()之后的一段时间内,关闭的套接字仍会占用分配给它们的本地地址。这取决于操作系统,但可能在几分钟左右。我尝试调用setReuseAddress(true)套接字,希望它的地址在close()被调用后可以立即重用。不幸的是,事实并非如此。

我创建套接字的代码是:

Socket socket = new Socket();
socket.setReuseAddress(true);
socket.connect(new InetSocketAddress(m_host, m_port));

但我仍然收到此错误:

java.net.BindException:地址已在使用中:过一会儿再连接。

还有其他方法可以完成我想做的事情吗?我想举个例子:打开100个套接字,将它们全部关闭,打开200个套接字,将它们全部关闭,打开300个,依此类推,最多可容纳2000个套接字。

任何帮助将不胜感激!


问题答案:

通过在两分钟的TIME_WAIT期间内打开许多出站套接字,就耗尽了出站端口的空间。您应该问自己的第一个问题是,这是否完全代表了现实的负载测试?真正的客户真的会这样做吗?如果没有,您只需要修改测试方法即可。

BTW SO_LINGER是 应用程序
在close()期间等待刷新数据的秒数。通常为零。如果这是发出关闭请求的结束时间,则该端口将在TIME_WAIT间隔内徘徊。这不是同一回事。可能会滥用SO_LINGER选项来解决此问题。但是,这也会在对等端引起异常行为,这又不是测试的目的。