首页 > 解决方案 > 在 Java 中查找空闲端口

问题描述

我需要在 Java 应用程序中找到一个空闲端口。为什么下面的代码不起作用?它总是返回 0。

public int findFreePort() {
    int portNumber = new InetSocketAddress(0).getPort();
    return portNumber;
}

Java doc ofnew InetSocketAddress()说:

创建一个套接字地址,其中 IP 地址是通配符地址,端口号是指定值。

有效端口值介于 0 和 65535 之间。端口号为零将使系统在绑定操作中选择一个临时端口。

标签: javaportip-addressjavadocinet-socket

解决方案


实际上,您输出的是您在构造函数中指定的端口成员。所以0是意料之中的。
javadoc 声明该端口将让系统为绑定操作选择一个临时端口。它并没有说明端口号将直接在InetSocketAddress实例中使用临时端口进行评估。实际上,您不会对这个InetSocketAddress实例执行绑定操作。
很明显,您没有打开/创建套接字通道来与此实例进行通信。所以你不能注意到端口的结果。

例如,这将 a 绑定ServerSocket到 a InetSocketAddress

ServerSocket ss = new ServerSocket(..);
ss.bind(new InetSocketAddress(0));

这是一个更完整的示例,说明了事情是如何工作的:

public class InetSockerAddressWithEphemeralPortMain {

    public static void main(String[] args) throws InterruptedException, IOException {
        InetSocketAddress randomSocketAddressFirst = new InetSocketAddress(0);

        try (ServerSocket ssOne = new ServerSocket()) {
            System.out.println("randomSocketAddress port before any binding : " + randomSocketAddressFirst.getPort());
            ssOne.bind(randomSocketAddressFirst);
            System.out.println("local port after first binding :" + ssOne.getLocalPort());
        }

        try (ServerSocket ssTwo = new ServerSocket()) {
            ssTwo.bind(randomSocketAddressFirst);
            System.out.println("local port after second binding :" + ssTwo.getLocalPort());
            System.out.println("randomSocketAddress port after all bindings : " + randomSocketAddressFirst.getPort());
        }

    }
}

输出 :

任何绑定前的 randomSocketAddress 端口:0

第一次绑定后的本地端口:65110

第二次绑定后的本地端口:65111

所有绑定后的 randomSocketAddress 端口:0

您可以看到InetSocketAddress对象始终保留0为端口值,而 ServerSocket对象受益于拾取的临时端口。


推荐阅读