干净地关闭rmi服务器

时间:2013-04-24 14:07:59

标签: java rmi

我在关闭服务器组件时遇到了一些麻烦,并希望得到一些帮助。

我的服务器代码如下所示,它有一个关闭服务器的方法

服务器

    private final String address = "127.0.0.1";
    private Registry registry;
    private int port = 6789;

    public RmiServer() throws RemoteException {
        try {
            registry = LocateRegistry.createRegistry(port);
            registry.rebind("rmiServer", this);
        } catch (RemoteException e) {
            logger.error("Unable to start the server. Exiting the application.", e);
            System.exit(-1);
        }
    }


    public void shutDownServer() throws RemoteException {
        int succesful = 0;
        try {
            registry.unbind("rmiServer");
            UnicastRemoteObject.unexportObject(this, true);
            Thread.sleep(1000);
        } catch (NotBoundException e) {
            logger.error("Error shutting down the server - could not unbind the registry", e);
            succesful = -1;
        } catch (InterruptedException e) {
            logger.info("Unable to sleep when shutting down the server", e);
            succesful = -1;
        }
        catch (AccessException e) {
            logger.info("Access Exception", e);
            succesful = -1;
        }
        catch (UnmarshalException e) {
            System.out.println(e.detail.getMessage());
            logger.info("UnMarshall Exception", e);
            succesful = -1;
        }
        catch (RemoteException e) {
            System.out.println(e.detail.getMessage());
            logger.info("Remote Exception", e);
            succesful = -1;
        }

        logger.info("server shut down gracefully");     
        System.exit(succesful);
}

我的客户端连接正常,没有问题,所以关闭我创建了一个新的应用程序,复制客户端代码进行连接,然后在服务器上调用close方法

关闭

公共课关闭{

private String serverAddress = "127.0.0.1";
private String serverPort = "6789";
private ReceiveMessageInterface rmiServer;
private Registry registry;

public Shutdown(){
    try {
        registry = LocateRegistry.getRegistry(serverAddress, (new Integer(serverPort)).intValue());
        rmiServer = (ReceiveMessageInterface) (registry.lookup("rmiServer"));
        logger.info("Client started correctly");
        rmiServer.shutDownServer();
        System.exit(0);
    }
    catch (UnmarshalException e ){
        logger.error("Unmarshall exception. Exiting application", e);
        System.exit(-1);
    }
    catch (RemoteException e) {
        logger.error("Remote object exception occured when connecting to server. Exiting application", e);
        System.exit(-1);
    } catch (NotBoundException e) {
        logger.error("Not Bound Exception occured when connecting to server. Exiting application", e);
        System.exit(-1);
    }
}

无论我尝试什么,我都会得到以下异常;

ERROR com.rmi.client.RMIClient - Unmarshall exception. Exiting application
java.rmi.UnmarshalException: Error unmarshaling return header; nested exception is: 
    java.net.SocketException: Connection reset
    at sun.rmi.transport.StreamRemoteCall.executeCall(Unknown Source)
    at sun.rmi.server.UnicastRef.invoke(Unknown Source)
    at java.rmi.server.RemoteObjectInvocationHandler.invokeRemoteMethod(Unknown Source)
    at java.rmi.server.RemoteObjectInvocationHandler.invoke(Unknown Source)
    at $Proxy0.shutDownServer(Unknown Source)
    at com.rmi.shutdown.Shutdown.<init>(Shutdown.java:31)
    at com.rmi.shutdown.Shutdown.main(Shutdown.java:52)
Caused by: java.net.SocketException: Connection reset
    at java.net.SocketInputStream.read(Unknown Source)
    at java.io.BufferedInputStream.fill(Unknown Source)
    at java.io.BufferedInputStream.read(Unknown Source)
    at java.io.DataInputStream.readByte(Unknown Source)
    ... 7 more

我相信这可能是因为客户端未正确断开连接并且只是“切断”但我不确定如何断开服务器端的连接?

请有人建议。

感谢

2 个答案:

答案 0 :(得分:5)

使用force = true的Unexport不会中止正在进行的调用。通常,它会让正在进行的调用运行完成。您的shutDownServer方法几乎是正确的,因为它取消注册远程引用并取消导出它。然而,它接下来的工作并不起作用。首先,它睡了一秒钟。这使呼叫保持正在进行并使客户端等待回复。然后关闭代码退出服务器JVM而不从远程调用返回。这会关闭客户端的连接,同时还在等待回复。这就是客户端获得连接重置异常的原因。

要干净地关闭,取消注册远程对象,使用force = true取消导出它(正如您已经完成的那样),然后返回。这将向客户端发送回复,让其远程调用完成,然后它将退出。回到服务器上,在完成最后一次正在进行的调用之后,如果没有其他对象被导出,并且如果没有其他任何东西保持JVM(例如非守护程序线程),则JVM将退出。您需要让RMI完成其服务器端处理,而不是调用System.exit()

答案 1 :(得分:1)

系统完全按照你的要求去做。你告诉它要取消自我,并将'force'参数设置为true,这会中止正在进行的调用,因此它会自动取消导出并中止正在进行的调用。只是忽略它,或者如果你坚持对关闭客户端做出干净的响应,让服务器为devport操作启动一个新的线程,并有一个短的延迟,以便关闭调用可以返回给客户端。