最近xx的一个项目用到了RMI,让我看一下这块,索性找了些资料学习了一下,与此同时顺便写一个入门小案例=。=
【RMI】:远程方法调用(Remote Method Invocation)是Enterprise JavaBeans的支柱,是建立分布式Java应用程序的方便途径。RMI是非常容易使用的,它非常的强大。
RMI的基础是接口,RMI构架基于一个重要的原理:接口和接口的具体实现是分开的。
下面通过具体的例子,建立一个简单的远程计算服务和使用它的客户程序 。
共有以下五个步骤:
1. 创建远程接口及声明远程方法(Compute.java)
2. 实现远程接口及远程方法(ComputeImpl.java)
3. 实现RMI注册服务,并注册远程对象(ComputeServer.java)
4. 客户端查找远程对象,并调用远程方法(ComputeClient)
5. 执行程序:启动服务ComputeServer;运行客户端ComputeClient进行调用
代码如下:
1.创建远程接口及声明远程方法
/**
* 创建远程接口,并声明远程方法
* 继承Remote类
*/
public interface Compute extends Remote {
//远程接口方法必须抛出RemoteException异常
public int add(int a,int b) throws RemoteException;
public int sub(int a,int b) throws RemoteException;
}
注意,这个接口继承自Remote,每一个定义的方法都必须抛出一个RemoteException异常对象。
2.实现远程接口及远程方法
/**
* 实现远程接口及远程方法
* 继承UnicastRemoteObject类
*/
public class ComputeImpl extends UnicastRemoteObject implements Compute {
/*
必须定义构造方法,即使是默认的构造方法,也必须创建出来,因为必须要抛出RemoteException异常
*/
protected ComputeImpl( ) throws RemoteException {
super();
}
@Override
public int add(int a,int b) throws RemoteException {
return a+b;
}
@Override
public int sub(int a,int b) throws RemoteException {
return a-b;
}
}
注意:这个实现类使用了继承UnicastRemoteObject去连接RMI系统。
事实上并不一定要这样做,如果一个类不是从UnicastRmeoteObject上继承,那必须使用它的exportObject()方法去联接到RMI。
如果一个类继承自UnicastRemoteObject,那么它必须提供一个构造函数并且声明抛出一个RemoteException对象。当这个构造函数调用了super(),它久激活UnicastRemoteObject中的代码完成RMI的连接和远程对象的初始化
有些资料可能会在这一步之后从接口实现类中生成桩(Stub)和框架(Skeleton)类文件,但是我通过运行发现,现在已过时,如图所示:
3. 实现RMI注册服务,并注册远程对象(ComputeServer.java)
/**
* 启动RMI注册服务并进行对象的注册
*/
public class ComputeServer {
public static void main(String[] args) {
try {
//通过LocateRegistry.createRegistry(1099)方式启动RMI注册服务
//指定端口为1099(默认的)
LocateRegistry.createRegistry(1099);
//创建远程对象的一个或多个实例
Compute c1 = new ComputeImpl();
//将c1对象注册到RMI注册服务器上,并且命名为c1(也就是map中的key)
Naming.rebind("c1",c1);
// 如果要把c1对象注册到另外一台启动了RMI注册服务的机器上,则写上ip地址
// Naming.rebind("//192.168.1.105:1099/c1",c1);
System.out.println("注册服务已经准备好了!");
} catch (RemoteException e) {
e.printStackTrace();
} catch (MalformedURLException e) {
e.printStackTrace();
}
}
}
4. 客户端查找远程对象,并调用远程方法(ComputeClient)
/**
* 客户端查找远程对象,并调用远程方法
*/
public class ComputeClient {
public static void main(String[] args) {
try {
//查找远程对象
Compute c1 = (Compute) Naming.lookup("c1");
// 若要从另外一台启动了RMI注册服务的机器上查找远程对象,则加上ip地址
// Compute hello2 = (Compute) Naming.lookup("//192.168.1.105/hello1");
//调用远程方法
//结果会得到这个方法返回回来的数据,
//而实际执行还是在另一端执行的
System.out.println(c1.add(6, 3));
System.out.println(c1.sub(5, 2));
} catch (NotBoundException e) {
e.printStackTrace();
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (RemoteException e) {
e.printStackTrace();
}
}
}
5.启动服务ComputeServer;运行客户端ComputeClient就可以实现调用
得到最终结果如图:
至此,就成功的创建了一个RMI系统,并且使之正常工作了=。=