1

アップデート:

問題は、一種の双方向接続を使用していることです。

クライアント側では、これは機能します:

String a = this.lobby.getMsg();

これはそうではありません:

this.lobby.login((Player)UnicastRemoteObject.exportObject(player, 0));

とアイデア?



ネットワーク通信に RMI を使用する Java アプリケーションに取り組んでいます。

1 台のコンピューターではすべて問題なく動作しますが、インターネット経由で実行しようとすると問題が発生しました。

クライアントはサーバーからレジストリを取得し、ロビー オブジェクトを検索できるようです。しかし、クライアントが rmi 経由でロビー オブジェクトのログイン メソッドを呼び出そうとすると、非常に時間がかかり、次の例外が発生します。

奇妙なことは、この例外に記載されている間違った IP (クライアント IP) があることです。サーバーと同じマシンでクライアントを実行すると、完全に機能します。

java.rmi.ServerException: RemoteException occurred in server thread; nested exception is: 
    java.rmi.ConnectException: Connection refused to host: <client ip(should be server ip?)>; nested exception is: 
    java.net.ConnectException: Die Wartezeit für die Verbindung ist abgelaufen (=time out)
    at sun.rmi.server.UnicastServerRef.dispatch(UnicastServerRef.java:353)
    at sun.rmi.transport.Transport$1.run(Transport.java:177)
    at sun.rmi.transport.Transport$1.run(Transport.java:174)
    at java.security.AccessController.doPrivileged(Native Method)
    at sun.rmi.transport.Transport.serviceCall(Transport.java:173)
    at sun.rmi.transport.tcp.TCPTransport.handleMessages(TCPTransport.java:553)
    at sun.rmi.transport.tcp.TCPTransport$ConnectionHandler.run0(TCPTransport.java:808)
    at sun.rmi.transport.tcp.TCPTransport$ConnectionHandler.run(TCPTransport.java:667)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1110)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:603)
    at java.lang.Thread.run(Thread.java:722)
    at sun.rmi.transport.StreamRemoteCall.exceptionReceivedFromServer(StreamRemoteCall.java:273)
    at sun.rmi.transport.StreamRemoteCall.executeCall(StreamRemoteCall.java:251)
    at sun.rmi.server.UnicastRef.invoke(UnicastRef.java:160)
    at java.rmi.server.RemoteObjectInvocationHandler.invokeRemoteMethod(RemoteObjectInvocationHandler.java:194)
    at java.rmi.server.RemoteObjectInvocationHandler.invoke(RemoteObjectInvocationHandler.java:148)
    at $Proxy0.einloggen(Unknown Source)
    at Client.login(Client.java:64)

関連するコードは次のとおりです。

私のサーバークラス:

    public static final int RMI_PORT = 55555;
    public static final String SERVER_IP = "xyz.com";

    /**
     * Startet Spielserver
     * 
     * @param args
     */
    public static void main(String[] args) {
        System.out.println("Server starten..");

        System.setProperty("java.rmi.server.hostname", SERVER_IP);
        try {
            if (System.getSecurityManager() == null) {
                System.setSecurityManager(new RMISecurityManager());
            }

            Registry registry = LocateRegistry.getRegistry(SERVER_IP, RMI_PORT);

            Lobby lobby = new LobbyImpl();
            UnicastRemoteObject.exportObject(lobby, 0);
            registry.bind("Lobby", lobby);
        } catch (RemoteException e) {
            e.printStackTrace();
        } catch (AlreadyBoundException e) {
            e.printStackTrace();
        }
    }

クライアントクラス:

public Client() {
        try {
            Registry registry = LocateRegistry.getRegistry("xyz.com", Server.RMI_PORT);
            lobby = (Lobby) registry.lookup("Lobby");
        } catch (RemoteException e) {
            e.printStackTrace();
        } catch (NotBoundException e) {
            e.printStackTrace();
        }
        this.gui = new GUI(this);
    }

    public void login(String name) throws RemoteException {
        Player player = new Player(name);
        this.lobby.login((Player)UnicastRemoteObject.exportObject(player, 0));  
    }

クライアント起動スクリプト:

#!/bin/bash
SCRIPTDIR=$(dirname $0)
java -classpath $SCRIPTDIR/bin Client

サーバー起動スクリプト:

#!/bin/bash
SCRIPTDIR=$(dirname $0)

cd ${SCRIPTDIR}/bin
rmiregistry 55555 &
cd - 
java -classpath $SCRIPTDIR/bin -Djava.security.policy=$SCRIPTDIR/src/server.policy -Djava.rmi.server.codebase=file:$SCRIPTDIR/bin/ Server
killall rmiregistry

最後になりましたが、最初の (テスト) ポリシー ファイル:

grant {
    permission java.security.AllPermission;
};

何が問題なのですか?

よろしく

4

2 に答える 2

2

RMI ホームページの RMI FAQ の項目 A.1 を参照する必要があります。ここでの問題は、間違った IP アドレスがリモート スタブに埋め込まれていることです。したがって、クライアントでターゲット IP アドレスを自分で指定しているため、ルックアップは機能しますが、リモート メソッドを実行すると、スタブ内の IP アドレスに依存しているため、これは間違っています。解決策は、/etc/hosts ファイルを修正するか、サーバー JVM でシステム プロパティ java.rmi.server.hostname を設定することです。

于 2012-06-27T12:00:57.503 に答える
0

エラーメッセージは

java.rmi.ConnectException: Connection refused to host: <wrong ip of the server>;

そのサーバーと telnet をサービス ポイントに ping できることを確認します。

于 2012-06-27T11:19:33.810 に答える