RMI (Remote Method Invocation) 模型是一种分布式对象应用,使用 RMI 技术可以使一个 JVM 中的对象,调用另一个 JVM 中的对象方法并获取调用结果。这里的另一个 JVM 可以在同一台计算机也可以是远程计算机。因此,RMI 意味着需要一个 Server 端和一个 Client 端。
Server 端通常会创建一个对象,并使之可以被远程访问。
这个对象被称为远程对象。Server 端需要注册这个对象可以被 Client 远程访问。
Client 端调用可以被远程访问的对象上的方法,Client 端就可以和 Server 端进行通信并相互传递信息。
正所谓 “知其然知其所以然”,在开始编写 RMI 代码之前,有必要了解一下 RMI 的工作原理,RMI 中 Client 端是和 Server 端是如何通信的呢?
下图的可以帮助我们理解RMI 的工作流程。
从图中可以看到,Client 端有一个被称 Stub 的东西,有时也会被成为存根,它是 RMI Client 的代理对象,Stub 的主要功能是请求远程方法时构造一个信息块,RMI 协议会把这个信息块发送给 Server 端。
这个信息块由几个部分组成:
远程对象标识符。
调用的方法描述。
编组后的参数值(RMI协议中使用的是对象序列化)。
既然 Client 端有一个 Stub 可以构造信息块发送给 Server 端,那么 Server 端必定会有一个接收这个信息快的对象,称为 Skeleton 。
它主要的工作是:
解析信息快中的调用对象标识符和方法描述,在 Server 端调用具体的对象方法。
取得调用的返回值或者异常值。
把返回值进行编组,返回给客户端 Stub.
到这里,一次从 Client 端对 Server 端的调用结果就可以获取到了。
1.定义传输的对象,传输的对象需要实现序列化(Serializable)接口。
?public class Emp implements Serializable {
? ? ?private Integer empNo;
? ? ?private String empName;
? ? ?private String addr;
? ? ?
? ? ?//省略 get/set
? ? ?@Override
? ? ?public String toString() {
? ? ? ? ?return "Emp{" +
? ? ? ? ? ? ? ? ?"empNo=" + empNo +
? ? ? ? ? ? ? ? ?", empName='" + empName + '\'' +
? ? ? ? ? ? ? ? ?", addr='" + addr + '\'' +
? ? ? ? ? ? ? ? ?'}';
? ? }
?}
Server 端主要是构建一个可以被传输的类 User,一个可以被远程访问的类 UserService,同时这个对象要注册到 RMI 开放给客户端使用。
2.定义服务器接口(需要继承 Remote 类,方法需要抛出 RemoteException)。
?public interface EmpService extends Remote {
? ? ?void addEmp(Emp emp) throws RemoteException;
? ? ?Emp findEmpById(Integer empNo) throws RemoteException;
??
?}
3.实现服务器接口(需要继承 UnicastRemoteObject 类,实现定义的接口)。
?
public class EmpServiceImpl extends UnicastRemoteObject implements EmpService {
? ? ?public EmpServiceImpl() throws RemoteException {
? ? ? ? ?super();
??
? ? }
??
? ? ?@Override
? ? ?public void addEmp(Emp emp) throws RemoteException {
? ? ? ? ?System.out.println(emp);
? ? ? ? ?System.out.println("保存数据成功。");
??
? ? }
??
? ? ?@Override
? ? ?public Emp findEmpById(Integer empNo) throws RemoteException {
? ? ? ? ?System.out.println("id:" + empNo);
? ? ? ? ?System.out.println("查询数据成功。");
? ? ? ? ?Emp emp = new Emp();
? ? ? ? ?emp.setEmpNo(empNo);
? ? ? ? ?emp.setEmpName("测试数据");
? ? ? ? ?emp.setAddr("测试数据");
? ? ? ? ?return emp;
??
? ? }
?}
4.注册( rmiregistry)远程对象,并启动服务端程序。
服务端绑定了 UserService 对象作为远程访问的对象,启动时端口设置为 2022。
?public class TestServer {
? ? ?public static void main(String[] args) throws Exception {
? ? ? ? ?EmpService imp = new EmpServiceImpl();
? ? ? ? ?//注册远程服务的端口
? ? ? ? ?LocateRegistry.createRegistry(2022);
? ? ? ? ?//将远程服务对象绑定为远程服务
? ? ? ? ?Naming.rebind("rmi://127.0.0.1:2022/a", imp);
? ? ? ? ?System.out.println("server启动成功。。。");
??
??
? ? }
?}
相比 Server 端,Client 端就简单的多。直接引入可远程访问和需要传输的类,通过端口和 Server 端绑定的地址,就可以发起一次调用。
?public class TestClient {
? ? ?public static void main(String[] args) throws Exception {
? ? ? ? ?EmpService server = (EmpService) Naming.lookup("rmi://127.0.0.1:2022/a");
? ? ? ? ?System.out.println("server:"+server);
? ? ? ? ?//远程方法调用
? ? ? ? ?Emp emp = new Emp();
? ? ? ? ?emp.setAddr("tj");
? ? ? ? ?emp.setEmpName("喵星兔");
? ? ? ? ?emp.setEmpNo(2021);
? ? ? ? ?server.addEmp(emp);
??
? ? ? ? ?System.out.println(server.findEmpById(100));
??
? ? }
?}
启动 Server 端。
server启动成功。。。
Emp{empNo=100, empName='测试数据', addr='测试数据'}