Java RMI

一、RMI需求
RMI允许不同的Java虚拟机之间进行对象间的通信,从而共享各个虚拟机上的资源和处理能力。RMI的目的是使分布在不同的虚拟机中的对象工作起来像是本地对象,以达到位置透明性,应用程序不用知道对象是本地还是远程的,屏蔽应用程序对底层的复杂实现细节。引用远程对象的JVM叫客户机,包含远程对象的JVM叫服务器。
二、RMI结构
如何使RMI开发者不用考虑调用目的对象是在本地还远程?通过本地、远程透明性的一种架构来完成,即存根(Stub)和骨架(Skeleton)。RMI在客户机端使用存根,存根就像服务器端代码的代理。客户通过存根在网络上查找所调用的远程对象。从客户机的角度来看,这些存根看起来完全像服务器端的方法,使用同样的方法名、参数列表和返回值,但不包含实际的方法代码,客户机取得的是Interface的reference,而无法取得实现该接口的class。
三、存根和骨架图

存根包含将客户端请求传送到服务器、处理从服务器传回的结果和将结果返给客户机的代码。
在服务器端,称为骨架的代理接收客户机的远程调用,将它们转换成在服务器上的本地调用,进行本地调用并将结果传回客户机。存根和骨架由RMI编译程序自动产生,它分析编写的服务器端类,生成存根和骨架的代码段。
四、RMI类库
RMI包括7个包:
(1)java.rmi :核心API。
(2)java.rmi.activation,可激活对象的API。
(3)java.rmi.dgc,分布式垃圾收集器(DGC)的API。
(4)java.rmi.registry,RMI网络命名服务的API。
(5)java.server,服务器端操作的API。
(6)java.rmi,RMI-IIOP的核心API,RMI-IIOP也是一种远程方法调用,在EJB中使用RMI-IIOP。
(7)java.rmi.CORBA,RMI-IIOP另一个轻便的API,为RMI-IIOP生成插头及连接线和RMI-IIOP运行时间之间提供标准接口。
五、定位远程对象
客户程序采用命名和目录服务找到远程服务,在编写客户程序时,必须告诉客户机提供命名服务的主机地址和端口号,通过JNDI服务,查找到远程对象。RMI命名服务为一个注册查询服务,提供远程服务的对象必须注册到RMI命名服务器上,在服务器上记录名字和远程对象的映射关系,这种关系以hashTable的结构存储。在注册表中,一个远程对象与一个名字联系起来,在客户程序想调用远程对象上的一个方法时,先要通过寻找名字得到对象的引用。寻找过程会返回一个对象的远程引用,即一个Stub。客户机代码得到一个Stub对象,它通过指定的服务器名称和对象名称来访问服务器对象。
六、RMI参数传递
(1)基本参数传递
当在远程方法中传递的参数或返回值是基本数据类型时,RMI就通过值进行传递。
(2)对象类型参数
采用值传递对象以及所引用的对象。
(3)远程对象参数
远程对象是采用传递引用的方式。
(4)对象序列化
将Java对象转换成对象的bit-blob表示,以便于在网络上传递、写入文件。对象序列化的方式实现java.io.Serializable接口,JVM虚拟机来完成对象序列化的具体实现。
(5)如何使用关键字transient
对象很大,通过网络传输或复制太浪费资源 远程对象的资源不可能在本地机器上进行重建,如数据库连接、线程对象等。对象中存在敏感信息,不希望通过网络传输
七、RMI-IIOP
(1)CORBA(Common Object Request Broker Architecture,公共对象请求代理体系结构) ,CORBA 采用传输协议为IIOP。CORBA 是独立于语言的 。CORBA 使用 CosNaming定位远程对象 。
(2)JRMP(Java Remote Messaging Protocol,Java 远程消息传递协议)作为RMI传输协议 。RMI 是纯粹 Java 到 Java 的。 RMI 使用 JNDI 定位远程对象 。
(3)RMI和CORBA的结合形成了RMI-IIOP
八、编写一个RMI应用程序
(1)实现RMI的步骤
<1>编写远程接口。这一接口将远程对象上有效的方法公布到RMI客户机上。它必须扩展java.rmi.Remote。
<2>编写远程对象。远程对象是实现远程接口的类。远程对象也可能包括用于内部操作和内部管理的其他方法,这些方法不通过接口公布。远程对象可以含有main方法,通过在main方法中编写远程对象的注册程序,实现把远程对象绑定到JNDI树上,以便远程客户访问远程对象。
<3>使用java编译程序编译远程接口和服务。
<4>编写调用远程对象的客户代码和使用java编译器编译客户代码。
<5>编写RMI安全文件
<6>通过RMIC编译器生成STUB和SKELETON。
<7>启动RMI注册服务器,START RMIREGISTRY
<8>向RMI注册服务器注册远程对象
<9>运行RMI客户端,获取远程对象提供的公布的方法。
(2) 编写远程接口
package rmiexample;

import java.rmi.*;

public interface HelloInterface extends Remote{

//这个方法被远程客户调用,被远程对象实现
public String sayHello() throws RemoteException;
}

注意:
1)这个接口只将一个方法sayHello公布到RMI客户机上,在远程接口中公布的方法必须抛出RemoteException。
2)远程接口必须扩展Remote这个接口,Remote接口的完整路径是java.rmi.Remote。
3)作为参数或返回值传递的一个远程对象必须声明为远程接口,不可声明为远程对象类。
(3) 编写远程对象

package rmiexample;
import java.rmi.*;

public class HelloServer extends UnicastRemoteObject
implements HelloInterface{
//这个方法被远程客户调用.
public String sayHello() throws RemoteException{
//向远程对象运行的虚拟机发送一个信息
System.out.println(“HelloServer.sayHello()”+”has been remotely invoked.”);
//向远程客户返回一个信息
Return “hello,remote world!”;

//main()方法实现把远程对象注册到RMI注册服务器上,以便远程客户端通过JNDI查找到远程对象。
Public static void main(String[] args){
/创建并安装安全管理器    if(System.getSecurityManager()==null)    {       System.setSecurityManager(new RMISecurityManager());    }
try{
//生成一个远程对象实例
HelloServer obj=new HelloServer();
//把实例注册到RMI注册服务器上
Naming.rebind(“HelloServer”,obj);
//向运行远程对象的JVM控制台发送一个信息
System.out.println(“HelloSrever was created and “+”bound in the registry “+”to the name HelloServer”);
}catch(Exception e){
System.out.println(“HelloServer error: “+e.getMessage());
e.printStackTrace();
}
}
}
注意:
1)这个最基本的服务类只实现了两个方法:在远程接口中定义的sayHello方法和在RMI注册表中注册服务的main方法。
2)Main方法创建一个远程对象实例,然后在HelloServer名字下使用命名服务注册它。
3)远程必须扩展UnicastRemoteObject类
4)实现远程接口,其返回类型、方法名、参数和抛出的例外必须和接口中的一致。
(4) 编译远程接口和远程对象
用javac编译远程接口和远程对象。
(5)产生远程对象存根和骨架
创建RemoteObject的主干和框架。要完成这个工作可使用rmic编译器,rmic编译器生成远程对象的存根和骨架。存根(Stub)是远程对象在客户端的代理,它将RMI调用传递给服务器端的骨架(Skeleton),后者负责将该调用传递给实际的远程方法
语法:
rmic –d d:\rmiexample rmiexample.HelloServer
该命令将编译生成HelloIterface_WLStub.class和HelloInterface_wlSkel.class并写到输出目录。
当RMI编译程序运行时,它为存根和骨架产生中间Java源文件,将那些源文件编译到类中,然后删除源文件。
注意:
在JDK1.2以上的版本中,Skeleton已经不再需要,只保留的远程客户端的Stub。
// Stub class generated by rmic, do not edit.
// Contents subject to change without notice.

package rmiexample;

public final class HelloServer_Stub
extends java.rmi.server.RemoteStub
implements rmiexample.HelloInterface, java.rmi.Remote
{
private static final long serialVersionUID = 2;

private static java.lang.reflect.Method $method_sayHello_0;

static {
try {
$method_sayHello_0 = rmiexample.HelloInterface.class.getMethod("sayHello", new java.lang.Class[] {});
} catch (java.lang.NoSuchMethodException e) {
throw new java.lang.NoSuchMethodError(
"stub class initialization failed");
}
}
// constructors
public HelloServer_Stub(java.rmi.server.RemoteRef ref) {
super(ref);
}

// methods from remote interfaces

// implementation of sayHello()
public java.lang.String sayHello()
throws java.rmi.RemoteException
{
try {
Object $result = ref.invoke(this, $method_sayHello_0, null, 6043973830760146143L);
return ((java.lang.String) $result);
} catch (java.lang.RuntimeException e) {
throw e;
} catch (java.rmi.RemoteException e) {
throw e;
} catch (java.lang.Exception e) {
throw new java.rmi.UnexpectedException("undeclared checked exception", e);
}
}
}
(6)编写远程调用服务的客户机
package rmiexample;
import java.rmi.*;
Public class HelloClient{
public final static String SREVERNAME=“rmi://202.196.9.175:1099/HelloServer”;
Public static void main(String[] args){
try{
HelloInterface obj=(HelloInterface)Naming.lookup(SERVERANME);
System.out.println(“Connected to HelloServer.”);
String message=obj.sayHello();
System.out.println(message);
}catch(Throwable t){
t.printStackTrace();
System.exit(-1);
}
}
}
(7)编译客户端程序
Javac rmiexample.HelloClient
(8)配置和启动RMI注册服务器
启动注册服务器:start rmiregistry [端口号]
(9)编辑RMI安全文件
grant {
permission java.security.AllPermission "", "";
};
(10)运行RMI服务
java –Djava.security.policy=“e:\test\rmiexample\policy.policy” rmiexample.HelloServer
Java –classpath %classpath% rmiexample.HelloClient

(11)运行客户机并测试服务
java –Djava.security.policy=“e:\test\rmiexample\policy.policy” rmiexample.HelloClient

Java –classpath %classpath% rmiexample.HelloClient
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值