从懵逼到恍然大悟之Java中RMI的使用

此处讲的是Java中的RMI,而不是通用意义上的RMI,关于通用的RMI可以参考分布式之RPC的协议以及错误处理 这篇文章。

一、Java RMI简介

Java RMI用于不同虚拟机之间的通信,这些虚拟机可以在不同的主机上、也可以在同一个主机上;一个虚拟机中的对象调用另一个虚拟上中的对象的方法,只不过是允许被远程调用的对象要通过一些标志加以标识。这样做的特点如下:

  • 优点:避免重复造轮子;
  • 缺点:调用过程很慢,而且该过程是不可靠的,容易发生不可预料的错误,比如网络错误等;

RMI中的核心是远程对象(remote object),除了对象本身所在的虚拟机,其他虚拟机也可以调用此对象的方法,而且这些虚拟机可以不在同一个主机上。每个远程对象都要实现一个或者多个远程接口来标识自己,声明了可以被外部系统或者应用调用的方法(当然也有一些方法是不想让人访问的)。

1.1 RMI的通信模型

从方法调用角度来看,RMI要解决的问题,是让客户端对远程方法的调用可以相当于对本地方法的调用而屏蔽其中关于远程通信的内容,即使在远程上,也和在本地上是一样的。

从客户端-服务器模型来看,客户端程序直接调用服务端,两者之间是通过JRMPJava Remote Method Protocol)协议通信,这个协议类似于HTTP协议,规定了客户端和服务端通信要满足的规范。

但是实际上,客户端只与代表远程主机中对象的Stub对象进行通信,丝毫不知道Server的存在。客户端只是调用Stub对象中的本地方法,Stub对象是一个本地对象,它实现了远程对象向外暴露的接口,也就是说它的方法和远程对象暴露的方法的签名是相同的。客户端认为它是调用远程对象的方法,实际上是调用Stub对象中的方法。可以理解为Stub对象是远程对象在本地的一个代理,当客户端调用方法的时候,Stub对象会将调用通过网络传递给远程对象。

java 1.2之前,与Stub对象直接对话的是Skeleton对象,在Stub对象将调用传递给Skeleton的过程中,其实这个过程是通过JRMP协议实现转化的,通过这个协议将调用从一个虚拟机转到另一个虚拟机。在Java 1.2之后,与Stub对象直接对话的是Server程序,不再是Skeleton对象了。

所以从逻辑上来看,数据是在ClientServer之间横向流动的,但是实际上是从ClientStub,然后从SkeletonServer这样纵向流动的。

这里写图片描述

1.2 重要的问题

1.2.1 数据的传递问题

我们都知道在Java程序中引用类型(不包括基本类型)的参数传递是按引用传递的,对于在同一个虚拟机中的传递时是没有问题的,因为的参数的引用对应的是同一个内存空间,但是对于分布式系统中,由于对象不再存在于同一个内存空间,虚拟机A的对象引用对于虚拟机B没有任何意义,那么怎么解决这个问题呢?

  • 第一种:将引用传递更改为值传递,也就是将对象序列化为字节,然后使用该字节的副本在客户端和服务器之间传递,而且一个虚拟机中对该值的修改不会影响到其他主机中的数据;但是对象的序列化也有一个问题,就是对象的嵌套引用就会造成序列化的嵌套,这必然会导致数据量的激增,因此我们需要有选择进行序列化,在Java中一个对象如果能够被序列化,需要满足下面两个条件之一:
    • Java的基本类型;
    • 实现java.io.Serializable接口(String类即实现了该接口);
    • 对于容器类,如果其中的对象是可以序列化的,那么该容器也是可以序列化的;
    • 可序列化的子类也是可以序列化的;
  • 第二种:仍然使用引用传递,每当远程主机调用本地主机方法时,该调用还要通过本地主机查询该引用对应的对象,在任何一台机器上的改变都会影响原始主机上的数据,因为这个对象是共享的;

RMI中的参数传递和结果返回可以使用的三种机制(取决于数据类型):

  • 简单类型:按值传递,直接传递数据拷贝;
  • 远程对象引用(实现了Remote接口):以远程对象的引用传递;
  • 远程对象引用(未实现Remote接口):按值传递,通过序列化对象传递副本,本身不允许序列化的对象不允许传递给远程方法;

1.2.2 远程对象的发现问题

在调用远程对象的方法之前需要一个远程对象的引用,如何获得这个远程对象的引用在RMI中是一个关键的问题,如果将远程对象的发现类比于IP地址的发现可能比较好理解一些。

在我们日常使用网络时,基本上都是通过域名来定位一个网站,但是实际上网络是通过IP地址来定位网站的,因此其中就需要一个映射的过程,域名系统(DNS)就是为了这个目的出现的,在域名系统中通过域名来查找对应的IP地址来访问对应的服务器。那么对应的,IP地址在这里就相当于远程对象的引用,而DNS则相当于一个注册表(Registry)。而域名在RMI中就相当于远程对象的标识符,客户端通过提供远程对象的标识符访问注册表,来得到远程对象的引用。这个标识符是类似URL地址格

阅读终点,创作起航,您可以撰写心得或摘录文章要点写篇博文。去创作
  • 64
    点赞
  • 252
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 12
    评论
Java RMI(Remote Method Invocation)是 Java 实现远程方法调用的一种机制,它允许一个 Java 程序从另一个 Java 程序调用对象。下面是一个简单的 Java RMI 的示例: 1. 定义接口 首先需要定义一个接口,用于客户端和服务端之间进行通信。比如: ``` public interface MyInterface extends Remote { public String sayHello() throws RemoteException; } ``` 这里的 MyInterface 继承了 Remote 接口,表示这个接口是一个远程接口。 2. 实现接口 接下来需要在服务端实现这个接口,比如: ``` public class MyInterfaceImpl extends UnicastRemoteObject implements MyInterface { public MyInterfaceImpl() throws RemoteException { super(); } public String sayHello() throws RemoteException { return "Hello, world!"; } } ``` 这里的 MyInterfaceImpl 类实现了 MyInterface 接口,并继承了 UnicastRemoteObject 类。这个类的构造函数需要抛出 RemoteException 异常。 3. 注册服务 在服务端需要将 MyInterfaceImpl 的实例注册到 RMI registry ,比如: ``` MyInterface myInterface = new MyInterfaceImpl(); Registry registry = LocateRegistry.getRegistry(); registry.bind("MyInterface", myInterface); ``` 这里使用了 LocateRegistry.getRegistry() 方法获取 RMI registry 的引用,然后使用 registry.bind() 方法将 MyInterfaceImpl 的实例注册到 RMI registry 。 4. 调用服务 在客户端需要获取 MyInterfaceImpl 的实例并调用它的方法,比如: ``` Registry registry = LocateRegistry.getRegistry("localhost"); MyInterface myInterface = (MyInterface) registry.lookup("MyInterface"); String result = myInterface.sayHello(); System.out.println(result); ``` 这里使用registry.lookup() 方法获取 MyInterfaceImpl 的实例,然后调用它的 sayHello() 方法并输出结果。 需要注意的是,使用 RMI 调用远程方法时,可能会抛出 RemoteException 异常,需要进行处理。此外,还需要在客户端和服务端都使用相同的接口和实现类,以保证通信的正确性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

lmy86263

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值