远程过程调用概述-RMI简介

简介:

RPC是远程过程调用(Remote Procedure Call)的缩写形式。SAP系统RPC调用的原理其实很简单,有一些类似于三层构架的C/S系统,第三方的客户程序通过接口调用SAP内部的标准或自定义函数,获得函数返回的数据进行处理后显示或打印。RPC采用客户机/服务器模式。请求程序就是一个客户机,而服务提供程序就是一个服务器。首先,客户机调用进程发送一个有进程参数的调用信息到服务进程,然后等待应答信息。在服务器端,进程保持睡眠状态直到调用信息到达为止。当一个调用信息到达,服务器获得进程参数,计算结果,发送答复信息,然后等待下一个调用信息,最后,客户端调用进程接收答复信息,获得进程结果,然后调用执行继续进行。

远程调用流程图:

 

工作原理:

运行时,一次客户机对服务器的RPC调用,其内部操作大致有如下十步:

1.调用客户端句柄;执行传送参数

2.调用本地系统内核发送网络消息

3.消息传送到远程主机

4.服务器句柄得到消息并取得参数

5.执行远程过程

6.执行的过程将结果返回服务器句柄

7.服务器句柄返回结果,调用远程系统内核

8.消息传回本地主机

9.客户句柄由内核接收消息

10.客户接收句柄返回的数据

远程调用的几种方式:

在java领域中知名的有:RMI、XML-RPC、Binary-RPC、SOAP、CORBA、JMS等。

1.RMI

     RMI是个典型的为java定制的远程通信协议, 我们都知道,在single vm中,我们能够通过直接调用java object instance来实现通信,那么在远程通信时,假设也能依照这样的方式当然是最好了。这样的远程通信的机制成为RPC(RemoteProcedure Call),RMI正是朝着这个目标而诞生的。传输的标准格式是Java Object Stream;基于Java串行化机制将请求的Java Object信息转化为流。传输协议是Socket。

2.XML-RPC

      XML-RPC也是一种和RMI类似的远程调用的协议,它和RMI的不同之处在于它以标准的xml格式来定义请求的信息(请求的对象、方法、參数 等),这种优点是在跨语言通讯的时候也能够使用。所以RMI与RPC的差别之中的一个是RPC是跨语言的。 传输的标准格式是XML。将XML转化为流。传输协议是HTTP。

3.Binary-RPC

 Binary-RPC和XML-RPC是差点儿相同,不同之处仅在于传输的标准格式由XML转为了二进制的格式。 传输的标准格式是二进制文件。将二进制文件转化为传输的流。传输协议是HTTP。

4.SOAP

 SOAP(SimpleObject Access Protocol),是一个用于分布式环境的、轻量级的、基于XML进行信息交换的通信协议,能够觉得SOAP是XML RPC的高级版。两者的原理全然同样。都是http+XML,不同的仅在于两者定义的XML规范不同。SOAP也是Webservice採用的服务调用协议标准。

5.CORBA

Common Object Request BrokerArchitecture(公用对象请求代理[调度]程序体系结构),是一组用来定义“分布式对象系统”的标准,由OMG(Object Menagement Group)作为发起和标准制定单位。CORBA的目的是定义一套协议,符合这个协议的对象能够互相交互,不论它们是用什么样的语言写的。不论它们执行于 什么样的机器和操作系统。CORBA是个类似于SOA的体系架构。涵盖可选的远程通信协议,但其本身不能列入通信协议。

6.JMS

      JMS是实现java领域远程通信的一种手段和方法。基于JMS实现远程通信时和RPC是不同的。尽管能够做到RPC的效果,但由于不是从协议 级别定义的。因此我们不觉得JMS是个RPC协议,但它确实是个远程通信协议,在其它的语言体系中也存在着类似JMS的东西,能够统一的将这类机制称为消 息机制。而消息机制呢,一般是高并发、分布式领域推荐的一种通信机制,这里的主要一个问题是容错。JMS注重的是消息交换,RMI注重的是对象方法调用,所以目的不同。JMS大多时候是异步的松耦合,RMI大多时候是同步的紧耦合。JMS规定的传输格式是Message,将參数信息放入Message中,传输协议不限。基于JMS也是经常使用的实现远程异步调用的方法之中的一个。

RMI介绍

  远程方法调用(RMI)顾名思义是一台机器上的程序调用另一台机器上的方法。这样可以大致知道RMI是用来干什么的,但是这种理解还不太确切。RMI是Java支撑分布式系统的基石,例如著名的EJB组件。RMI是远程过程调用(RPC)的一种面向对象实现,RMI底层是通过socket通信和对象序列化技术来实现的。这里引用Wikipedia对RMI的介绍:

RMI基本原理

  RMI的目的就是要使运行在不同的计算机中的对象之间的调用表现得像本地调用一样。RMI 应用程序通常包括两个独立的程序:服务器程序和客户机程序。RMI 需要将行为的定义与行为的实现分别定义, 并允许将行为定义代码与行为实现代码存放并运行在不同的 JVM 上。在 RMI 中, 远程服务的定义是存放在继承了 Remote 的接口中。远程服务的实现代码存放在实现该定义接口的类中。RMI 支持两个类实现一个相同的远程服务接口: 一个类实现行为并运行在服务器上, 而另一个类作为一个远程服务的代理运行在客户机上。客户程序发出关于代理对象的调用方法, RMI 将该调用请求发送到远程 JVM 上, 并且进一步发送到实现的方法中。实现方法将结果发送给代理, 再通过代理将结果返回给调用者。

RMI 构建三个抽象层, 高层覆盖低层, 分别负责Socket通信, 参数和结果的序列化和反序列化等工作。存根( Stub) 和骨架( Skeleton) 合在一起形成了 RMI 构架协议。下面的引用层被用来寻找各自的通信伙伴, 在这一层还有一个提供名字服务的部分, 称为注册表( registry) 。最下一层是传输层, 是依赖于 TCP/IP 协议实现客户机与服务器的互联。

       当客户端调用远程对象方法时, 存根负责把要调用的远程对象方法的方法名及其参数编组打包,并将该包向下经远程引用层、传输层转发给远程对象所在的服务器。通过 RMI 系统的 RMI 注册表实现的简单服务器名字服务, 可定位远程对象所在的服务器。该包到达服务器后, 向上经远程引用层, 被远程对象的 Skeleton 接收, 此 Skeleton 解析客户包中的方法名及编组的参数后, 在服务器端执行客户要调用的远程对象方法, 然后将该方法的返回值( 或产生的异常) 打包后通过相反路线返回给客户端, 客户端的 Stub 将返回结果解析后传递给客户程序。事实上, 不仅客户端程序可以通过存根调用服务器端的远程对象的方法, 而服务器端的程序亦可通过由客户端传递的远程接口回调客户端的远程对象方法。在分布式系统中, 所有的计算机可以是服务器, 同时又可以是客户机。

RMI应用示例

      Remote 接口用于标识其方法可以从非本地虚拟机上调用的接口。任何远程对象都必须直接或间接实现此接口。只有在“远程接口”(扩展 java.rmi.Remote 的接口)中指定的这些方法才可远程使用。 也就是说需要远程调用的方法必须在扩展Remote接口的接口中声名并且要抛出RemoteException异常才能被远程调用。远程对象必须实现java.rmi.server.UniCastRemoteObject类,这样才能保证客户端访问获得远程对象时,该远程对象将会把自身的一个拷贝序列化后以Socket的形式传输给客户端,此时客户端所获得的这个拷贝称为“存根”,而服务器端本身已存在的远程对象则称之为“骨架”。其实此时的存根是客户端的一个代理,用于与服务器端的通信,而骨架也可认为是服务器端的一个代理,用于接收客户端的请求之后调用远程方法来响应客户端的请求。 远程对象的接口和实现必须在客户端和服务器端同时存在并且保持一致才行。

代码实现:

HelloDefine.java

1 package com.etc;
2 
3 import  java.rmi.Remote;
4 import  java.rmi.RemoteException;
5 //远程接口定义
6 public interface HelloDefine extends Remote {
7     public String helloWorld() throws RemoteException;
8     public String sayHello(String name) throws RemoteException; 
9 }

HelloDefineImp.java

 1 package com.etc;
 2 
 3 import  java.rmi.RemoteException;
 4 import  java.rmi.server.UnicastRemoteObject;
 5 //远程接口实现
 6 public class HelloDefineImp extends UnicastRemoteObject implements HelloDefine {
 7     private static final long serialVersionUID = 1L;
 8     public HelloDefineImp() throws RemoteException {
 9         super();
10     }
11     public String helloWorld() throws RemoteException {
12         return "Hello AlphaGo!";
13     }
14     public String sayHello(String name) throws RemoteException {
15         return "Hello" + name +"!";
16     }
17 }

HelloServer.java

 1 package com.etc;
 2 
 3 import  java.net.MalformedURLException;
 4 import  java.rmi.AlreadyBoundException;
 5 import  java.rmi.Naming;
 6 import  java.rmi.RemoteException;
 7 import  java.rmi.registry.LocateRegistry;
 8 //服务端绑定
 9 public class HelloServer {
10     HelloDefine hello;
11     public void server() throws RemoteException, MalformedURLException, AlreadyBoundException {
12         hello = new HelloDefineImp(); 
13         //远程对象注册表实例
14         LocateRegistry.createRegistry(8888);
15         //把远程对象注册到RMI注册服务器上
16         Naming.bind("rmi://localhost:8888/Hello", hello);
17         System.out.println("server:对象绑定成功!");
18     }
19 }

HelloClient.java

 1 package com.etc;
 2 
 3 import  java.net.MalformedURLException;
 4 import  java.rmi.Naming;
 5 import  java.rmi.NotBoundException;
 6 import  java.rmi.RemoteException;
 7 //客户端调用
 8 public class HelloClient {
 9     public HelloDefine hello;
10     public void client() throws MalformedURLException, RemoteException, NotBoundException {
11         //在RMI注册表中查找指定对象
12         hello = (HelloDefine) Naming.lookup("rmi://localhost:8888/Hello");
13         //调用远程对象方法
14         System.out.println("client:");
15         System.out.println(hello.helloWorld());
16         System.out.println(hello.sayHello("起飞吧骚年!"));
17     }
18 }

 转载内容来源:https://www.cnblogs.com/wxisme/p/5296441.html

 

转载于:https://www.cnblogs.com/weekstart/p/10733962.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值