本文章共分为三部分
- RMI简介
- RMI原理
- RMI使用
- RMI实战
下面详细介绍
一 RMI简介
远程方法调用(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使用
项目url图四 RMI实战
GoldbachDefine
import java.rmi.Remote;
import java.rmi.RemoteException;
public interface GoldbachDefine extends Remote{
public String checkNum(int num) throws RemoteException;
}
GoldbachDefineImp
import java.rmi.RemoteException;
import java.rmi.server.UnicastRemoteObject;
public class GoldbachDefineImp extends UnicastRemoteObject implements GoldbachDefine {
public GoldbachDefineImp() throws RemoteException {
// TODO Auto-generated constructor stub
super();
}
@Override
public String checkNum(int num) throws RemoteException {
// TODO Auto-generated method stub
for(int i=2;i<(num/2+1);i++){
if(isPrimeNum(i)&&isPrimeNum(num-i)){
return "the two number is "+i+"and "+(num-i);
}
}
return 2333333+"";
}
public Boolean isPrimeNum(int num){
for(int i=2;i<(num/2+1);i++){
if(num%i==0){
return false;
}
}
return true;
}
}
GoldbachServer
import java.net.MalformedURLException;
import java.rmi.AlreadyBoundException;
import java.rmi.Naming;
import java.rmi.RemoteException;
import java.rmi.registry.LocateRegistry;
public class GoldbachServer {
public static void main(String args[])throws RemoteException,MalformedURLException,AlreadyBoundException{
GoldbachDefine goldbach;
goldbach=new GoldbachDefineImp();
//register object
LocateRegistry.createRegistry(8888);
//bing the object to server
Naming.bind("rmi://localhost:8888/Goldbach", goldbach);
System.out.println("server:begin to listen");
}
}
GoldbachClient
import java.net.MalformedURLException;
import java.rmi.Naming;
import java.rmi.NotBoundException;
import java.rmi.RemoteException;
public class GoldbachClient {
public static void main(String args[])throws MalformedURLException,RemoteException,NotBoundException{
//find register object
GoldbachDefine goldbach = (GoldbachDefine) Naming.lookup("rmi://localhost:8888/Goldbach");
//output the client result
System.out.println("client:");
System.out.println(goldbach.checkNum(8));
}
}
本文章参考以下文章
维基百科-RMI远程方法调用
远程方法调用(RMI)原理与示例
RMI远程方法调用
以上就是RMI的全部啦。大家加油~
网友评论