我想启动一个侦听端口的服务器。我可以显式地指定端口,它工作。但我想以自动的方式找到一个端口。在这方面,我有两个问题。

我应该在哪个范围内搜索端口号?(我使用的端口是12345、12346和12347,没有问题)。 我如何知道给定的端口是否被其他软件占用?


当前回答

这可能对你没有太大帮助,但在我的(Ubuntu)机器上,我有一个文件/etc/services,其中至少给出了一些应用程序使用/保留的端口。这些是这些应用的标准端口。

不能保证这些端口正在运行,只是这些应用程序使用的默认端口(所以如果可能的话,你不应该使用它们)。

定义了500多个端口,大约一半是UDP,一半是TCP。

这些文件是使用IANA的信息制作的,请参见IANA指定的端口号。

其他回答

如果您将0作为端口号传递给ServerSocket的构造函数,它将为您分配一个端口。

如果您需要在范围内使用:

public int nextFreePort(int from, int to) {
    int port = randPort(from, to);
    while (true) {
        if (isLocalPortFree(port)) {
            return port;
        } else {
            port = ThreadLocalRandom.current().nextInt(from, to);
        }
    }
}

private boolean isLocalPortFree(int port) {
    try {
        new ServerSocket(port).close();
        return true;
    } catch (IOException e) {
        return false;
    }
}

如果你正在使用Spring框架,最直接的方法是:

private Integer laancNotifyPort = SocketUtils.findAvailableTcpPort();

你也可以设置一个可接受的范围,它会在这个范围内搜索:

private Integer laancNotifyPort = SocketUtils.findAvailableTcpPort(9090, 10090);

这是一种方便的方法,它抽象了复杂性,但在内部与本线程中的许多其他答案相似。

你可以让ServerSocket为你找到一个端口,然后关闭它:

private int getFreePort() {
        var freePort = 0;
        try (ServerSocket s = new ServerSocket(0)) {
            freePort = s.getLocalPort();
        } catch (Exception e) {
            e.printStackTrace();
        }

        return freePort;
    }

根据维基百科,如果你不需要一个“众所周知”的端口,你应该使用端口49152到65535。

判断一个端口是否正在使用的唯一方法是尝试打开它。