【问题标题】:MulticastSocket not responding after failureMulticastSocket 失败后没有响应
【发布时间】:2010-10-09 16:42:33
【问题描述】:

尝试在 MulticastSocket 上调用 joinGroup(addr) 时出现 SocketException。这只发生在我们设置为在机器启动时自动启动应用程序的 Windows 机器上。

似乎抛出异常是因为 Windows 尚未完全完成其启动过程,这里是异常。

java.net.SocketException: error setting options

    at java.net.PlainDatagramSocketImpl.join(Native Method)

    at java.net.PlainDatagramSocketImpl.join(Unknown Source)

    at java.net.MulticastSocket.joinGroup(Unknown Source)

在我们的应用启动时,如果我们在尝试加入群组之前等待一分钟,一切正常。

所以我们决定设置一个重试循环,以便它在网络可用时立即连接,这似乎可以工作。两次失败后,第三次尝试加入群组成功。

问题是,现在 MulticastSocket 没有收到来自该组的任何消息,即使它加入得很好。

每次失败后我都会创建一个新的 MulticastSocket 并丢弃旧的。

为什么在一个 MulticastSocket 上加入组失败会影响没有任何错误加入的组,我该如何解决这个问题?

【问题讨论】:

    标签: java windows multicastsocket


    【解决方案1】:

    我从来没有弄清楚为什么套接字在成功加入组后不会收到消息。不过,我确实想出了一个变通办法。

    我遍历所有网络接口并确保列表中有一个有效的接口并且它已启动并正在运行。我要做的下一件事是尝试在 MulticastSocket 上设置该网络接口。如果这些测试通过,那么我让套接字尝试加入组。它似乎有效,但我仍然想了解更多关于幕后发生的事情。

    private void validateNetworkInterfaces() throws IOException {
    
        Enumeration nis = NetworkInterface.getNetworkInterfaces();
        List<NetworkInterface> nics = new ArrayList<NetworkInterface>();
        while (nis.hasMoreElements()) {
            NetworkInterface ni = (NetworkInterface) nis.nextElement();
    
            logger.debug("nic name: " + ni.getDisplayName());
            logger.debug("nic isLoopback(): " + ni.isLoopback());
            logger.debug("nic isPointToPoint(): " + ni.isPointToPoint());
            logger.debug("nic isVirtual(): " + ni.isVirtual());
            logger.debug("nic isUp(): " + ni.isUp());
            logger.debug("nic supportsMulticast(): " + ni.supportsMulticast());
    
            if (!ni.isLoopback() && !ni.isPointToPoint() && !ni.isVirtual() && ni.isUp() && ni.supportsMulticast()) {
                logger.debug("adding nic: " + ni.getDisplayName());
                nics.add(ni);               
            }               
    
        }
    
        //check to make sure at least one network interface was found that supports multicast.
        if (nics.size() == 0) throw new SocketException("No network interfaces were found that support multicast.");
    
        //make sure the network interface can be set on a multicast socket
        for (NetworkInterface nic : nics) {
            logger.debug("attempting to set network interface on nic: " + nic.getDisplayName());
            MulticastSocket ms1 = new MulticastSocket(45599);
            ms1.setNetworkInterface(nic);
        }
    
    }
    

    【讨论】:

      【解决方案2】:

      我知道这是旧的,但可靠的多播答案似乎很少见。

      我认为你会更好:

      final InetAddress localHost = InetAddress.getLocalHost();
      final NetworkInterface networkInterface = NetworkInterface.getByInetAddress(localHost);
      

      因为这更简洁,并且还可以确保获得将实际接收多播消息的 NIC。

      【讨论】:

        猜你喜欢
        • 1970-01-01
        • 2021-07-03
        • 2019-02-04
        • 1970-01-01
        • 2010-12-09
        • 1970-01-01
        • 2017-01-04
        • 1970-01-01
        • 1970-01-01
        相关资源
        最近更新 更多