Java RMI对客户端本身和来自服务器的远程调用的客户端变量有不同的实例化

时间:2014-11-21 10:21:46

标签: java scope rmi

我们正在尝试使用Javas RMI创建一个系统。问题是无法使用Java RMI从服务器访问客户端上的维护列表。似乎RMI连接正在处理初始化列表的副本。

下面是一个使用整数的最小示例,客户端每秒递增一次,直到它等于10.服务器一直收到0。

任何人都知道我们做错了什么?

只需将服务器和客户端作为Java应用程序运行。

ServerDefaultImpl.java

package rmi;

import java.rmi.RemoteException;
import java.rmi.registry.LocateRegistry;
import java.rmi.registry.Registry;
import java.rmi.server.UnicastRemoteObject;

public class ServerDefaultImpl implements EIServerRemote, Runnable {
    ClientRemote client;
    private boolean running = true;

    public ServerDefaultImpl() {
        try {
            LocateRegistry.createRegistry(Registry.REGISTRY_PORT);

            ServerDefaultImpl server = this;
            EIServerRemote stub = (EIServerRemote) UnicastRemoteObject.exportObject(server, 0);

            Registry registry = LocateRegistry.getRegistry();
            registry.rebind("test", stub);
        } catch (RemoteException e) {
            e.printStackTrace();
        }
        new Thread(this).start();
    }

    public static void main(String[] args) {
        new ServerDefaultImpl();
    }

    @Override
    public void run() {
        while (true == running) {
            try {
                Thread.sleep(1000);
                if (null != client) { //Client not connected yet.
                    int test = client.test();
                    System.out.println(test);
                    running = test <= 10;
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    @Override
    public void attachClientListener(ClientRemote client) throws RemoteException {
        this.client = client;
    }
}

EIServerRemote.java

package rmi;

import java.rmi.Remote;
import java.rmi.RemoteException;

public interface EIServerRemote extends Remote {
    void attachClientListener(ClientRemote client) throws RemoteException;
}

ClientRemote.java

package rmi;

import java.io.Serializable;
import java.rmi.Remote;

public interface ClientRemote extends Remote,Serializable {
    int test();
}

ClientDefaultImpl.java

package rmi;

import java.rmi.NotBoundException;
import java.rmi.RemoteException;
import java.rmi.registry.LocateRegistry;
import java.rmi.registry.Registry;

public class ClientDefaultImpl implements Runnable,
        ClientRemote {

    private static final long serialVersionUID = 4846141863099303590L;

    protected EIServerRemote server = null;

    public int test;

    public boolean running = true;

    public ClientDefaultImpl(String serverName) {
        test = 0;
        try {
            connect(serverName);
        } catch (RemoteException | NotBoundException e) {
            e.printStackTrace();
        }
        new Thread(this).start();
    }

    public static void main(String[] args) {
        new ClientDefaultImpl("test");
    }

    public void connect(String serverName) throws RemoteException,
            NotBoundException {
        Registry registry = LocateRegistry.getRegistry();
        EIServerRemote s = (EIServerRemote) registry.lookup(serverName);
        server = s;
        s.attachClientListener((ClientRemote) this);
    }

    @Override
    public void run() {
        while (true == running) {
            try {
                Thread.sleep(1000);
                System.out.println(test++);
                running = test <= 10;
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    @Override
    public int test() {
        return test;
    }
}

1 个答案:

答案 0 :(得分:1)

  

似乎RMI连接正在处理初始化列表的副本。

这是对的。该列表不是远程对象,因此它通过序列化传递和返回。