6

如何防止 Java 应用程序绑定到另一个进程已在 Windows 上绑定的套接字?

我有一个问题,我有一个正在侦听端口 80 的 Java 应用程序。该应用程序启动正常并且没有报告异常。我不知道为什么我无法连接端口 80。其他端口工作正常。我检查了 netstat 是否有其他在 80 上监听的进程,并找到了 Skype。我不认为这是可能的,但经过一些研究,我猜 Skype 正在使用 SO_REUSEADDR 选项进行监听。在这种状态下,接受申请是不确定的。我希望我的 Java 应用程序在此实例中失败并出现绑定异常(或其他)。

如果我可以通过 Java 访问该选项,我似乎可以使用 SO_EXCLUSIVEADDRUSE,但我认为这是不可能的。SO_REUSEADDR 周围有很多问题和答案,但我找不到回答我的问题的答案。这不仅仅是关于 Skype(我可以关闭监听部分),我希望我的程序在这种情况下更加健壮。

这是来自 Windows 7 机器上的 netstat -abn 的片段:

    Proto  Local Address          Foreign Address        State
    TCP    0.0.0.0:80             0.0.0.0:0              LISTENING [java.exe]
    TCP    0.0.0.0:80             0.0.0.0:0              LISTENING [Skype.exe]

这就是为什么我假设 Skype 正在使用 SO_REUSEADDR

这些进程似乎没有在不同的接口上监听。

这是代码片段。端口为 80:

    myServerSocket = new ServerSocket(myTcpPort);
    while (true) {
        new HTTPSession( myServerSocket.accept(), threadPool );
    }

作为进一步的信息,我创建了一个示例程序,以尽量减少任何副作用或错误处理的消息。

    import java.net.ServerSocket;

    public class PortTest
    {
        public static void main(String[] args)
        {
            System.out.println( "Hit Ctrl-C to stop.\n" );

            try {
                ServerSocket myServerSocket = new ServerSocket(80);
                System.out.println( "Before the accept() call.");
                myServerSocket.accept();
                System.out.println( "After the accept() call." );
            }
            catch (Exception ex ) {
                System.out.println("Error listening.");
                ex.printStackTrace();
            }
         }

    }

运行此示例程序 (PortTest) 以及 Skype 正在运行并侦听端口 80 时,我仍然没有遇到异常。为了进一步测试,我执行了第二个实例 PortTest,我确实看到了正在使用的端口消息。

    Error listening.
    java.net.BindException: Address already in use: JVM_Bind
            at java.net.DualStackPlainSocketImpl.bind0(Native Method)
            at java.net.DualStackPlainSocketImpl.socketBind(Unknown Source)
            at java.net.AbstractPlainSocketImpl.bind(Unknown Source)
            at java.net.PlainSocketImpl.bind(Unknown Source)
            at java.net.ServerSocket.bind(Unknown Source)
            at java.net.ServerSocket.<init>(Unknown Source)
            at java.net.ServerSocket.<init>(Unknown Source)
            at PortTest.main(PortTest.java:10)
4

2 回答 2

2
socket = new ServerSocket(0);

会自动为你选择一个空闲端口。

此外,此代码将告诉您端口是否可用:

boolean portAvaliable = true;
ServerSocket s = null;
try {
   s = new ServerSocket(yourPort);
}
catch (IOException e) {
   portAvaliable = false;
} 
finally {
   if (s != null)
      try {
        s.close();
      } 
      catch (IOException e) { 
         //handle the exception
      }
}

检查布尔值portAvaliable以识别端口状态。

于 2013-05-16T15:51:44.310 回答
0

对于任何想知道的人,7u25 之前的 Java 都有这个问题。

以下是Java 7u25 发行说明的摘录:

Windows 平台上网络 API 实现的变化

Windows 上的网络 API 的实现已更改为SO_EXCLUSIVEADDRUSE默认使用套接字选项。此更改对于解决在使用需要绑定到同一端口的 IPv4 和 IPv6 应用程序时出现的异常情况是必要的。

这个问题是 Windows 独有的,因为[1] SO_REUSEADDR[2]在那里工作的方式很时髦。

于 2018-03-31T16:24:15.947 回答