RMI NotSerializableException虽然它是一个远程对象

Gra*_*uin 9 java rmi

我正在编写一个基于RMI的小型聊天应用程序.

这个想法是:客户端在服务器上注册自己,每次服务器从客户端收到消息时,他都会将此消息推送给所有其他客户端.

但是我收到NotSerializableException,但是,我作为方法参数传递的对象实现了Remote接口.

这是一些代码:(有问题的部分是(ClientChat Implementation)中的this参数this.chatServ.registriereClient(this);)

(ClientChat)界面:

public interface ChatClient extends Remote
{

}
Run Code Online (Sandbox Code Playgroud)

(ClientChat)执行:

public class ChatClientImpl implements ChatClient
{

    ChatServer chatServ;
    String clientName;

    public ChatClientImpl(String clientName, ChatServer chatServ) {
        this.chatServ = chatServ;
        this.clientName = clientName;
        try {
            this.chatServ.registriereClient(this);
        } catch (RemoteException e) {
            e.printStackTrace();
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

(ServerChat)接口

public interface ChatServer extends Remote
{
        void registriereClient(ChatClient client) throws RemoteException;

}
Run Code Online (Sandbox Code Playgroud)

(ServerChat)实现

public class LobbyChatServerImpl implements ChatServer
{

    ArrayList<ChatClient> clientListe = null;

    @Override
    public void registriereClient(ChatClient client) {
        System.out.println("Client registriert");
        this.clientListe.add(client);
    }
}
Run Code Online (Sandbox Code Playgroud)

客户:

  public static void main(String[] args) {
        ChatServer lobbyChatServer = null;
        try {
            Registry registry = LocateRegistry.getRegistry(Server.RMI_PORT);
            lobbyChatServer = (ChatServer) registry.lookup("LobbyChatServer");

        } catch (RemoteException e) {
            e.printStackTrace();
        } catch (NotBoundException e) {
            e.printStackTrace();
        }

        ChatClient lobbyChat = new ChatClientImpl(name, lobbyChatServer);
  }
Run Code Online (Sandbox Code Playgroud)

服务器:

    public static void main(String[] args) {
        try {
            if (System.getSecurityManager() == null) {
                System.setSecurityManager(new RMISecurityManager());
            }

            Registry registry = LocateRegistry.getRegistry(RMI_PORT);

            ChatServer lobbyChatStub = (ChatServer)UnicastRemoteObject.exportObject(new LobbyChatServerImpl(), 0);
            registry.bind("LobbyChatServer", lobbyChatStub);

        } catch (RemoteException e) {
            e.printStackTrace();
        } catch (AlreadyBoundException e) {
            e.printStackTrace();
        }
    }
Run Code Online (Sandbox Code Playgroud)

例外:

java.rmi.MarshalException: error marshalling arguments; nested exception is: 
    java.io.NotSerializableException: de.XX.Chat.ChatClientImpl
    at sun.rmi.server.UnicastRef.invoke(UnicastRef.java:156)
    at java.rmi.server.RemoteObjectInvocationHandler.invokeRemoteMethod(RemoteObjectInvocationHandler.java:194)
    at java.rmi.server.RemoteObjectInvocationHandler.invoke(RemoteObjectInvocationHandler.java:148)
    at $Proxy0.registriereClient(Unknown Source)
    at de.XX.Chat.ChatClientImpl.<init>(ChatClientImpl.java:19)
    at de.XX.Client.main(Client.java:49)
Caused by: java.io.NotSerializableException: de.XX.Chat.ChatClientImpl
    at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1180)
    at java.io.ObjectOutputStream.writeObject(ObjectOutputStream.java:346)
    at sun.rmi.server.UnicastRef.marshalValue(UnicastRef.java:292)
    at sun.rmi.server.UnicastRef.invoke(UnicastRef.java:151)
    ... 5 more
Run Code Online (Sandbox Code Playgroud)

如前所述,我想知道为什么我会得到这种异常,虽然ChatClientImpl已经是Remote.

希望你能帮我:)

use*_*421 14

作为参数传递的对象或远程方法的结果必须是:

  1. 可序列化(或可外部化),或

  2. 导出的远程对象.

你的不是.但是,因为它明确地实现了远程接口(2).扩展的对象UnicastRemoteObject在构造时自动导出.必须通过显式导出不需要的对象UnicastRemoteObject.exportObject().