一 .RMI概述
RMI(Remote Method Invocation)
RMI是分布式对象软件包,它简化了在多台计算机上的
JAVA应用之间的通信。必须在jdk1.1以上
RMI用到的类
什么是
RMI
使用这种机制,某一台计算机上的对象在调用另外一台计算机上的方法时,使用的程
序语法规则和在本地机上对象间的方法调用的语法规则一样。
优点
这种机制给分布计算的系统设计、编程都带来了极大的方便。
只要按照 RMI规则设计程序,可以不必再过问在 RMI之下的网络细节了,如:TCP和Socket等等。
任意两台计算机之间的通讯完全由 RMI负责。调用远程计算机上的对象就像本地对象一样方便。
只要按照 RMI规则设计程序,可以不必再过问在 RMI之下的网络细节了,如:TCP和Socket等等。
任意两台计算机之间的通讯完全由 RMI负责。调用远程计算机上的对象就像本地对象一样方便。
1、面向对象:
RMI可将完整的对象作为参数和返回值进行传递,而不仅仅是预定义的数据类型。
也就是说,可以将类似 Java哈西表这样的复杂类型作为一个参数进行传递。
也就是说,可以将类似 Java哈西表这样的复杂类型作为一个参数进行传递。
2、可移动属性:
RMI可将属性从客户机移动到服务器,或者从服务器移动到客户机。
3、设计方式:
对象传递功能使您可以在分布式计算中充分利用面向对象技术的强大功能,如二层和三层结构系统。
如果用户能够传递属性,那么就可以在自己的解决方案中使用面向对象的设计方式。
所有面向对象的设计方式无不依靠不同的属性来发挥功能,如果不能传递完整的对象——包括实现和类型
——就会失去设计方式上所提供的优点。
如果用户能够传递属性,那么就可以在自己的解决方案中使用面向对象的设计方式。
所有面向对象的设计方式无不依靠不同的属性来发挥功能,如果不能传递完整的对象——包括实现和类型
——就会失去设计方式上所提供的优点。
4、安全性:
RMI使用
Java内置的安全机制保证下载执行程序时用户系统的安全。
RMI使用专门为保护系统免遭恶意小程序侵害而设计的安全管理程序。
RMI使用专门为保护系统免遭恶意小程序侵害而设计的安全管理程序。
5、便于编写和使用
RMI使得
Java远程服务程序和访问这些服务程序的
Java客户程序的编写工作变得轻松、简单。
远程接口实际上就是 Java接口。
为了实现 RMI的功能必须创建远程对象任何可以被远程调用的对象必须实现远程接口。但远程
接口本身并不包含任何方法。因而需要创建一个新的接口来扩展远程接口。
新接口将包含所有可以远程调用的方法。远程对象必须实现这个新接口,由于新的接口扩展了
远程接口,实现了新接口,就满足了远程对象对实现远程接口的要求,所实现的每个对象都将
作为远程对象引用。
远程接口实际上就是 Java接口。
为了实现 RMI的功能必须创建远程对象任何可以被远程调用的对象必须实现远程接口。但远程
接口本身并不包含任何方法。因而需要创建一个新的接口来扩展远程接口。
新接口将包含所有可以远程调用的方法。远程对象必须实现这个新接口,由于新的接口扩展了
远程接口,实现了新接口,就满足了远程对象对实现远程接口的要求,所实现的每个对象都将
作为远程对象引用。
个人总结:
下面我们就来写一个
RMI的程序:
一.创建RMI程序的6个步骤:
1、定义一个远程接口的接口,该接口中的每一个方法必须声明它将产生一个RemoteException异常。
2、定义一个实现该接口的类。
3、使用 RMIC程序生成远程实现所需的残根和框架。
4、创建一个服务器,用于发布2中写好的类。
1、定义一个远程接口的接口,该接口中的每一个方法必须声明它将产生一个RemoteException异常。
2、定义一个实现该接口的类。
3、使用 RMIC程序生成远程实现所需的残根和框架。
4、创建一个服务器,用于发布2中写好的类。
5. 创建一个客户程序进行
RMI调用。
6、启动 rmiRegistry并 运行自己的远程服务器和客户程序。
6、启动 rmiRegistry并 运行自己的远程服务器和客户程序。
二. 程序详细说明
1.定义一个远程接口的接口,该接口中的每一个方法必须声明它将产生一个RemoteException异常。
- import
java.rmi.Remote; - import
java.rmi.RemoteException; - public
interface I_Hello extends java.rmi.Remote //需要从Remote继承 - {
-
public String SayHello() throws RemoteException; //需要抛出remote异常 - }
import java.rmi.Remote;
import java.rmi.RemoteException;
public interface I_Hello extends java.rmi.Remote //需要从Remote继承
{
public String SayHello() throws RemoteException; //需要抛出remote异常
}
抛出这个异常的原因
由于任何远程方法调用实际上要进行许多低级网络操作,因此网络错误可能在调用过程中随时发生。
因此,所有的 RMI操作都应放到try-catch块中。
2、定义一个实现该接口的类。
-
import java.io.PrintStream; - import
java.rmi.*; - import
java.rmi.server.UnicastRemoteObject; -
- public
class Hello extends UnicastRemoteObject //必须从UnicastRemoteObject 继承 -
implements I_Hello - {
-
public Hello() throws RemoteException //需要一个抛出Remote异常的默认初始化方法 -
{ -
} -
-
public String SayHello() //这个是实现I_Hello接口的方法 -
{ -
return "Hello world !!"; -
} - }
import java.io.PrintStream;
import java.rmi.*;
import java.rmi.server.UnicastRemoteObject;
public class Hello extends UnicastRemoteObject //必须从UnicastRemoteObject 继承
implements I_Hello
{
public Hello() throws RemoteException //需要一个抛出Remote异常的默认初始化方法
{
}
public String SayHello() //这个是实现I_Hello接口的方法
{
return "Hello world !!";
}
}
实现接口的类必须继承UnicastRemoteObject类。
扩展 java. rmi.server.UnicastRemoteObject
UnicastRemoteObject顾名思义,是让客户机与服务器对象实例建立一对一的连接。
扩展 java. rmi.server.UnicastRemoteObject
UnicastRemoteObject顾名思义,是让客户机与服务器对象实例建立一对一的连接。
3、使用RMIC程序生成远程实现所需的残根Stub 和 框架。
参考:
在
RMI中,客户机上生成的调动调用参数和反调动返回值的代码称为残根。有的书上称这部分代码为“主干”。
服务器上生成的反调动调用参数和进行实际方法调用调动返回值的代码称为框架。
生成残根和框架的工具
Rmic命令行工具( RMI Compiler)
格式:
Rmic classname
服务器上生成的反调动调用参数和进行实际方法调用调动返回值的代码称为框架。
生成残根和框架的工具
Rmic命令行工具( RMI Compiler)
格式:
Rmic classname
4、创建一个服务器,用于发布2中写好的类。
- import
java.rmi.*; - public
class RMI_Server - {
-
public static void main(String[] args) -
{ -
try -
{ -
Hello hello = new Hello(); //实例化要发布的类 -
Naming.rebind("RMI_Hello", hello); //绑定RMI名称 进行发布 -
System.out.println("=== Hello server Ready === "); -
} -
catch(Exception exception) -
{ -
exception.printStackTrace(); -
} -
} - }
import java.rmi.*;
public class RMI_Server
{
public static void main(String[] args)
{
try
{
Hello hello = new Hello(); //实例化要发布的类
Naming.rebind("RMI_Hello", hello); //绑定RMI名称 进行发布
System.out.println("=== Hello server Ready === ");
}
catch(Exception exception)
{
exception.printStackTrace();
}
}
}
5. 创建一个客户程序进行RMI调用。
- import
java.rmi.*; - public
class RMI_Client { -
public static void main(String[] args) { -
try -
{ -
I_Hello hello = (I_Hello) Naming.lookup("RMI_Hello"); //通过RMI名称查找远程对象 -
System.out.println(hello.SayHello()); //调用远程对象的方法 -
} catch (Exception e) -
{ -
e.printStackTrace(); -
} -
} -
- }
import java.rmi.*;
public class RMI_Client {
public static void main(String[] args) {
try
{
I_Hello hello = (I_Hello) Naming.lookup("RMI_Hello"); //通过RMI名称查找远程对象
System.out.println(hello.SayHello()); //调用远程对象的方法
} catch (Exception e)
{
e.printStackTrace();
}
}
}
Naming.lookup("
RMI_Hello") 其中的参数“
RMI_Hello”只是针对本机的
RMI查找,如果是异地的
RMI调用请参照
rmi://127.0.0.1:1099/
RMI_Hello
端口1099是默认的
RMI端口,如果你启动
rmiregistry 的时候(见第6点)没有指定特殊的端口号,默认就是1099
到此 我们 所有的代码编写都完成了,不过不要急着去
运行,请跟随第6点去
运行,因为
rmi 调用还会遇到一些特别的情况,偶花了牛劲,才找到原因的,许多刚用
RMI的人,常常被这些问题搞得吐血
6、启动rmiRegistry并运行自己的远程服务器和客户程序。
蓝字部分指定了stub类的路径。
2) 客户端的
运行
在文件最后加入下面代码:
此代码,开放了端口的connect访问权限
注意 你应该修改服务器那台机子的安全
配置文件,也就是你
运行
rmiregistry 和
RMI_Server的机子
另外,很多人修改完以后,仍然报这个错误,多数情况是由于你没有修改到正确的jdk 下的文件,而是修改到其他jdk的文件, 我们安装oracle ,
Weblogic等等软件的时候都会自带一个 jdk,他们会自动在操作系统的环境变量里面 加入jdk的路径,所以,你先要确定你
运行服务器端程序是用哪个jdk,再修改这个jdk下的
配置文件,确定当前jdk的路径很简单
开始 -》
运行-》
rmiregistry 看看这个DOS窗口标题 的路径,就是你当前系统默认jdk的路径了
客户端正常
运行以后,就会出现以下结果:
Hello world !!
这些字符是通过
RMI调用远程服务器的类返回的结果