我在关闭服务器组件时遇到了一些麻烦,并希望得到一些帮助。
我的服务器代码如下所示,它有一个关闭服务器的方法
服务器
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
我相信这可能是因为客户端未正确断开连接并且只是“切断”但我不确定如何断开服务器端的连接?
请有人建议。
感谢
答案 0 :(得分:5)
使用force = true的Unexport不会中止正在进行的调用。通常,它会让正在进行的调用运行完成。您的shutDownServer
方法几乎是正确的,因为它取消注册远程引用并取消导出它。然而,它接下来的工作并不起作用。首先,它睡了一秒钟。这使呼叫保持正在进行并使客户端等待回复。然后关闭代码退出服务器JVM而不从远程调用返回。这会关闭客户端的连接,同时还在等待回复。这就是客户端获得连接重置异常的原因。
要干净地关闭,取消注册远程对象,使用force = true取消导出它(正如您已经完成的那样),然后返回。这将向客户端发送回复,让其远程调用完成,然后它将退出。回到服务器上,在完成最后一次正在进行的调用之后,如果没有其他对象被导出,并且如果没有其他任何东西保持JVM(例如非守护程序线程),则JVM将退出。您需要让RMI完成其服务器端处理,而不是调用System.exit()
。
答案 1 :(得分:1)
系统完全按照你的要求去做。你告诉它要取消自我,并将'force'参数设置为true,这会中止正在进行的调用,因此它会自动取消导出并中止正在进行的调用。只是忽略它,或者如果你坚持对关闭客户端做出干净的响应,让服务器为devport操作启动一个新的线程,并有一个短的延迟,以便关闭调用可以返回给客户端。