Java 分布式处理技术

来源:互联网 发布:2017淘宝造物节店铺 编辑:程序博客网 时间:2024/05/20 13:05

转载
标签: 

杂谈

分类: Java

1.1 RMI 的基本概念

1.1.1 什么是RMI

RMI(Remote Method Invocation) 远程方法调用是一种计算机之间对象互相调用对方函数,启动对方进程的一种机制,使用这种机制,某一台计算机上的对象在调用另外一台计算机上的方法时,使用的程序语法规则和在本地机上对象间的方法调用的语法规则一样。

1.1.2 RMI 的用途

1、 分布式体系结构

我们为什么要使用分布式计算呢?

Ø         当我们想与多个用户或客户机共享一个中央资源(如一个数据库)时,就会使用分布式计算。

Ø         分布式计算用来利用多个系统的组合计算能力,以便比在单个系统上更有效或更快地解决问题。

可以用多种方法配置多个计算机系统以共享处理,包括共享内存、共享磁盘或只是共享一条公共通信通道。最新的技术允许物理上相隔很远的系统能够在处理计算问题时协同工作。

关于利用计算能力这一主题,因特网及伴随的通信协议 TCP/IP 的出现已使无数的计算机系统史无前例地连接起来。对一些应用程序来说,能够利用如此多的计算功能来解决问题是令人满意的。甚至更吸引人的是,大多数计算机系统都有充足的空闲时间,可以帮助解决其它问题。将来,网格计算会利用分布式计算能力进行出售,这与电力行业出售电能非常相似。

2、 Java 分布式对象编程技术

RMI 是 Enterprise JavaBeans 的支柱,是建立分布式 Java 应用程序的方便途径。只要按照 RMI 规则设计程序,可以不必再过问在 RMI 之下的网络细节了,如: TCP 和 Socket 等等。任意两台计算机之间的通讯完全由 RMI 负责。调用远程计算机上的对象就像本地对象一样方便。

1.1.3 RMI 应用程序分类

依据 RMI 应用程序各部分职责,可对应用程序进行如下分类:

Ø         服务器程序:服务器程序将创建多个远程对象,并使每个对象能够被引用。等待客户端调用创建好的远程对象上的方法。

Ø         客户端程序:从服务端程序中得到一个或多个远程对象的引用。客户端能用此引用调用远程对象上的方法。

Ø         对等计算程序:双方地位相等,互为对方的服务器和客户端。

1.2 创建 RMI 应用程序步骤

1、 定义远程接口

在 Java 中,远程对象是实现远程接口的类的实例 , 远程接口声明每个要远程调用的方法。在需要创建一个远程对象的时候,我们通过传递一个接口来隐藏基层的实施细节,客户通过接口句柄发送消息即可。远程接口具有如下特点:

Ø         远程接口必须为 public 属性。如果不这样,除非客户端与远程接口在同一个包内,否则 当试图装入实现该远程接口的远程对象时,调用会得到错误结果。

Ø         远程接口必须扩展接口 java.rmi.Remote 。

Ø         除与应用程序本身特定的例外之外,远程接口中的每个方法都必须在自己的 throws 从句中 声明 java.rmi.RemoteException 。(或 RemoteException 的父类)。

代码范例 1

package com.itjob;

import java.rmi.*;

 

public interface RmiSample extends Remote{

       public int sum(int a,int b) throws RemoteException;

      

}

 

2、 实现远程接口

远程对象实现类必须扩展远程对象 java.rmi.UnicastRemoteObject 类,并实现所定义的远程接口。远程对象的实现类中包含实现每个远程接口所指定的远程方法的代码。这个类也可以含有附加的方法,但客户只能使用远程接口中的方法。因为客户是指向接口的一个句柄,而不是它的哪个类。必须为远程对象定义构造函数,即使只准备定义一个默认构造函数,用它调用基础类构造函数。因为基础类构造函数可能会抛出 java.rmi.RemoteException ,所以即使别无它用必须抛出 java.rmi.RemoteException 例外。

代码范例 2

package com.itjob.rmi;

import java.rmi.*;

import java.rmi.server.*;

import com.itjob.RmiSample ;

public class RmiSampleImpl extends UnicastRemoteObject implements RmiSample{

    // 覆盖默认构造函数并抛出 RemoteException

       public RmiSampleImpl() throws RemoteException{

              super();

       }

    // 所有远程实现方法必须抛出 RemoteException

       public int sum(int a,int b) throws RemoteException{

              return a+b;

       }

}

 

3、 编写服务器类

包含 main 方法的类可以是实现类自身,也可以完全是另一个类。下面通过 RmiSampleServer 来创建一个远程对象的实例,并通过 java.rmi.registry.LocateRegistry 类的 createRegistry 方法从指定端口号启动注册服务程序,也可以通过执行 rmiregistry 命令启动注册服务程序,注册服务程序的缺省运行端口为 1099 。

代码范例 3

package com.itjob.rmi;

 

import java.rmi.*;

import java.rmi.registry.*;

 

public class RmiSampleServer{

       public static void main(String[] args){

               

              try{

                     LocateRegistry.createRegistry(8808);

                     RmiSampleImpl server=new RmiSampleImpl();

                     Naming.rebind("//localhost:8808/SAMPLE-SERVER",server);

                     System.out.println (" 远程对象注册成功, RMI 服务已经启动,等待客户端调用 ....");

              }catch(java.net.MalformedURLException me){

                     System.out.println ("Malformed URL:"+me.toString());

              }catch(RemoteException re){

                     System.out.println ("Remote exception:"+re.toString());

              }catch(AlreadyBoundException abe){

            System.out.println (" (AlreadyBound exception:"+ abe .toString());

}

       }

}

 

Ø         代码范例 3 中将将远程对象名字绑定到对远程对象的引用上:

LocateRegistry.createRegistry(8808); 指定本 RMI 服务程序不使用默认端口 1099, 而是使用自己指定的端口 8808 。

Naming.rebind("//localhost:8808/SAMPLE-SERVER" , Server); 将远程对象在服务器上注册并指定了将查找远程对象引用的 URL , URL 格式为 //host:port/name 。其中 host 是注册表所在的主机(远程或本地), port 是注册表接受调用的端口号, name 是未经注册表解释的简单字符串。 host 和 port 两者都是可选项。如果省略了 host ,则主机默认为本地主机。如果省略了 port ,则端口默认为 1099 ,该端口是 RMI 的注册表 rmiregistry 使用的“著名”端口。

代码范例 3 的运行结果如图所示:

图 14.1 代码范例 3 运行结果

 

4、 编写使用远程服务的客户机类

客户机类的主要功能有两个,一是通过 Naming.lookup 方法来构造注册服务程序 stub 程序实例,二是调用服务器远程对象上的远程方法。

代码范例 4

package com.itjob.rmi;

import java.rmi.*;

  import java.rmi.server.*;

  public class RmiSampleClient {

     public static void main(String[] args)

     {

         try {

             String url = "//localhost:8808/SAMPLE-SERVER";

            RmiSample RmiObject = (RmiSample)Naming.lookup(url);

            System.out.println(" 1 + 2 = " + RmiObject.sum(1,2) );

         } catch (RemoteException exc) {

             System.out.println("Error in lookup: " + exc.toString());

         } catch (java.net.MalformedURLException exc) {

             System.out.println("Malformed URL: " + exc.toString());

         } catch (java.rmi.NotBoundException exc) {

             System.out.println("NotBound: " + exc.toString());

         }

      }

  }

 

5、 为远程对象实现创建根和干

客户端是通过 Naming.lookup 方法来构造注册服务程序 stub 程序实例,通过该实例的引用来发起对远程对象方法调用的,所以在运行运行客户端应用前必须为远程对象实现创建根 (stub) 和干 (Skeleton) 。要创建存根程序和骨架文件,应以包含远程对象实现的已编译类包全名运行 rmic 编译器。存根( Stub )是远程对象在客户端的代理,它将 RMI 调用传递给服务器端的骨架( Skeleton ),后者负责将该调用传递给实际的远程方法。在命令行模块下运行 RMIC 调用:

图 14. 2 RMIC 命令

调用 RMIC 命令运行结果如图所示:

图 14.3 RMIC 运行结果

Ø         我们可以看系统帮我们生成了存根 (stub)RmiSampleImpl_Stub.class 但系统并没有帮我们生成骨架( Skeleton ) RmiSampleImpl_ Skeleton .class 。这与 JDK 的版本有关:

采用 JDK1.4 版本运行 rmic com.itjob.rmi. RmiSampleImpl 命令系统将会生成存根 (stub)RmiSampleImpl_Stub.class 和骨架( Skeleton ) RmiSampleImpl_ Skeleton .class 两个类文件;

采用 JDK1.5 版本运行 rmic com.itjob.rmi. RmiSampleImpl 命令系统将只会生成存根 (stub)RmiSampleImpl_Stub.class ,骨架( Skeleton ) RmiSampleImpl_ Skeleton .class 的功能将通反射技术由系统在运行时自动实现;

6、 运行程序

依次做完上述步骤后,我们现在来运行一下我们的 RMI 应用。先运行服务端程序,运行结果如图 14.1 所示。

接下来我们运行客户端程序(代码范例 4 ),运行结果如图所示:

图 14.4 客户端程序运行结果

看到上面结果说明我们客户端程序进行 RMI 远程调用已经成功了。

1.3 RMI 接口和类简介

负责指定 rmi 系统远程对象行为的接口和类在 java.rmi 包中定义的,接下来我们了解一下几个核心接口和类:

1、 Java.rmi.Remote 接口

在 rmi 中,远程接口声明了可从远程 java 虚拟机中调用的方法集,远程接口必须满足下列条件:

Ø         远程接口必须至少直接或间接的扩展 java.rmi.Remote 接口。

Ø         远程接口中的方法申明必须满足:远程方法申明在其 throws 子句中除了要包含与应用程序有关的异常之外,还必须包括 RemoteException 异常 ( 或她的父类 ) ;在远程方法申明中,作为参数或返回值申明的远程对象必须申明为远程接口,而非该接口的实现类。

2、 Java.rmi.RemoteException 类

RemoteException 类是在远程方法调用期间由 RMI 运行所抛出的异常,在使用了 rmi 系统的应用程序中,远程接口中申明的远程方法在其 throws 子句中必须指定 RemoteException 或者其超类。

Ø         当远程方法调用由于某种原因失败时,将抛出 RemoteException 异常。

Ø         RemoteException 类是一个已检验的异常,而不是 RuntimeException 。

3、 Java.rmi.server.RemoteObject 类

Ø         RMI 服务器函数由 RemoteObject 类及其子类 Remote


原创粉丝点击