有没有办法确定远程RMI服务器计算机是否是多宿主,即使其某些IP地址不可见?

时间:2012-05-09 16:39:47

标签: java sockets rmi

我遇到了几个障碍,让RMI服务器在多宿主机器上正常工作。我已经编写了自定义RMISocketFactory,并且我尝试了几种不同的方法来实现客户端createSocket(String host, int port)方法。一种方法仅在RMI服务器机器是多宿主时才起作用,另一种方式仅在RMI服务器机器不是时才起作用。以下是我的客户端createSocket实现中RMISocketFactory方法版本的代码:

仅在RMI服务器不是多宿主时才有效:

@Override
public Socket createSocket(String host, int port) throws IOException{
    Socket s = new Socket(host,port);

    //Do some book-keeping here ...

    return s;
}

仅在RMI服务器为多宿主时才有效:

private TreeSet<String> validHosts = null;

@Override
public Socket createSocket(String host, int port) throws IOException{
    Socket s = null;

    try {
        s = new Socket(host,port);
        synchronized(this){
            if(validHosts == null) validHosts = new TreeSet<String>();
            validHosts.add(host);
        }
    } catch (IOException e) {}

    if(s == null && validHosts != null){
        synchronized(this){
            for(String h : validHosts){
                try {
                    s = new Socket(h,port);
                } catch (IOException e) {}
            }
        }
    }

    if(s == null){
        try {
            String h = RemoteServer.getClientHost();
            s = new Socket(RemoteServer.getClientHost(),port);

            synchronized(this){
                if(validHosts == null) validHosts = new TreeSet<String>();
                validHosts.add(h);
            }
        } catch (Exception e) {}
    }

    //No dice - throw an exception:
    if(s == null){
        TreeSet<String> set = new TreeSet<String>();
        set.add(host+"(orig)");

        synchronized(this){
            if(validHosts != null) set.addAll(validHosts);
        }

        StringBuilder sb = new StringBuilder(
            "Could not connect socket on port "+port+
            " to any of the following hosts: ");
        for(String h : set) sb.append(h+" ");

        throw new IOException(sb.toString());
    }

    //Do some book-keeping here ...

    return s;
}

问题

似乎如果我能以某种方式判断服务器端机器是否是多宿主,那么我可以将这两组代码包装在if(serverIsMultihomed) ... else ...类型的语句中。我有能力在此调用中获取服务器的实际计算机名称,但是调用InetAddress.getAllByHostName(host)实际上并不返回所有主机的IP地址,只有那些对于客户端计算机上的NIC。因此,当使用实际上对客户端不可见的IP地址调用createSocket(host,port)时,这就是我遇到问题的地方。此外,将java.server.rmi.hostname设置为我要使用的服务器的IP地址并不能解决此问题,并且基本上禁止服务器连接到的不同子网上的计算机建立链接。 / p>

缺点是,还有另一种方法可以从不同的机器上获取所有主机的IP地址(可见或其他方式),还是我在这里完全咆哮错误的树?

2 个答案:

答案 0 :(得分:1)

你看过this Multihomed solution吗?

答案 1 :(得分:0)

如果服务器有一个对每个人都可见的IP地址,这似乎是Sun考虑的唯一多宿主案例,那么您需要将java.rmi.server.codebase设置为服务器JVM上的该地址。根本不需要套接字工厂。