Java RMI (Remote Method Invocation 遠程方法調用)是用Java在JDK1.1中實現的,它大大增強了Java開發分佈式應用的能力。Java作爲一種風靡一時的網絡開發語言,其巨大的威力就體現在它強大的開發分佈式網絡應用的能力上,而RMI就是開發百分之百純Java的網絡分佈式應用系統的核心解決方案之一。其實它可以被看作是RPC的Java版本。但是傳統RPC並不能很好地應用於分佈式對象系統。而Java RMI 則支持存儲於不同地址空間的程序級對象之間彼此進行通信,實現遠程對象之間的無縫遠程調用。RMI目前使用Java遠程消息交換協議JRMP(Java Remote Messaging Protocol)進行通信。JRMP是專爲Java的遠程對象制定的協議。因此,Java RMI具有Java的"Write Once,Run Anywhere"的優點,是分佈式應用系統的百分之百純Java解決方案。用Java RMI開發的應用系統可以部署在任何支持JRE(Java Run Environment Java,運行環境)的平臺上。但由於JRMP是專爲Java對象制定的,因此,RMI對於用非Java語言開發的應用系統的支持不足。不能與用非Java語言書寫的對象進行通信。本文擬從程序的角度舉例介紹怎樣利用RMI實現Java分佈式應用。 |
Java與.NET都提供了遠程處理功能,但不完全相同.Java遠程處理是通過一個“共享接口”實現的,而.NET可以通過一個“共享命令集”實現。下面就這兩種方式來具體說明。
Java 遠程處理
Java遠程方法調用(RMI)提供了Java程序語言的遠程通訊功能,這種特性使客戶機上運行的程序可以調用遠程服務器上的對象,使Java編程人員能夠在網絡環境中分佈操作。
創建一個簡單的Java分佈式遠程方法調用程序可以按以下幾個步驟操作,
一、定義遠程接口:
在 Java 中,遠程對象是實現遠程接口的類的實例, 遠程接口聲明每個要遠程調用的方法。在需要創建一個遠程對象的時候,我們通過傳遞一個接口來隱藏基層的實施細節,客戶通過接口句柄發送消息即可。
遠程接口具有如下特點:
1) 遠程接口必須爲public屬性。如果不這樣,除非客戶端與遠程接口在同一個包內,否則 當試圖裝入實現該遠程接口的遠程對象時,調用會得到錯誤結果。
2) 遠程接口必須擴展接口java.rmi.Remote。
3) 除與應用程序本身特定的例外之外,遠程接口中的每個方法都必須在自己的throws從句中 聲明java.rmi.RemoteException。(或 RemoteException 的父類)。
4) 作爲參數或返回值傳遞的一個遠程對象(不管是直接,還是本地對象中嵌入)必須聲明爲遠 程接口,而不應聲明爲實施類。
下面是遠程接口的接口RmiSample的定義
- package com.robin.demo.rmi.interf;
- import java.rmi.Remote;
- import java.rmi.RemoteException;
- public interface RmiSample extends Remote {
- public int sum(int a, int b) throws RemoteException;
- }
二、實現遠程接口:
遠程對象實現類必須擴展遠程對象java.rmi.UnicastRemoteObject類,並實現所定義的遠程接口。遠程對象的實現類中包含實現每個遠程接口所指定的遠程方法的代碼。這個類也可以含有附加的方法,但客戶只能使用遠程接口中的方法。因爲客戶是指向接口的一個句柄,而不是它的哪個類。必須爲遠程對象定義構造函數,即使只准備定義一個默認構造函數,用它調用基礎類構造函數。因爲基礎類構造函數可能會拋出 java.rmi.RemoteException,所以即使別無它用必須拋出java.rmi.RemoteException例外。
以下是遠程對象實現類的聲明:
- package com.robin.demo.rmi.impl;
- import java.rmi.RemoteException;
- import java.rmi.server.UnicastRemoteObject;
- import com.robin.demo.rmi.interf.RmiSample;
- public class RmiSampleImpl extends UnicastRemoteObject implements RmiSample {
- /**
- *
- */
- private static final long serialVersionUID = 2742977636753958461L;
- public RmiSampleImpl() throws RemoteException {
- super();
- }
- public int sum(int a, int b) throws RemoteException {
- return a + b;
- }
- }
三、編寫服務器類:
包含 main 方法的類可以是實現類自身,也可以完全是另一個類。下面通過RmiSampleServer 來創建一個遠程對象的實例,並通過java.rmi.registry.LocateRegistry類的createRegistry 方法從指定端口號啓動註冊服務程序,也可以通過執行 rmiregistry 命令啓動註冊服務程序,註冊服務程序的缺省運行端口爲 1099。必須將遠程對象名字綁定到對遠程對象的引用上: Naming.rebind("//localhost:8808/SAMPLE-SERVER"
, Server);
以下是服務器類的聲明:
- package com.robin.demo.rmi.server;
- import java.net.MalformedURLException;
- import java.rmi.Naming;
- import java.rmi.RemoteException;
- import java.rmi.registry.LocateRegistry;
- import com.robin.demo.rmi.impl.RmiSampleImpl;
- public class RmiSampleServer {
- /**
- * @param args
- */
- public static void main(String[] args) {
- try{
- LocateRegistry.createRegistry(8808);
- RmiSampleImpl server= new RmiSampleImpl();
- Naming.rebind("//localhost:8808/SAMPLE-SERVER" , server);
- }catch (MalformedURLException me){
- System.out.println("Malformed URL: " + me.toString());
- }catch(RemoteException re){
- System.out.println("Remote Exception: "+re.toString());
- }
- }
- }
四、編寫使用遠程服務的客戶機類:
客戶機類的主要功能有兩個,一是通過Naming.lookup方法來構造註冊服務程序 stub 程序實例,二是調用服務器遠程對象上的遠程方法。
以下是服務器類的聲明:
- package com.robin.demo.rmi.client;
- import java.rmi.Naming;
- import java.rmi.RemoteException;
- import com.robin.demo.rmi.interf.RmiSample;
- public class RmiSampleClient {
- /**
- * @param args
- */
- 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 rex) {
- System.out.println("Error in lookup: " + rex.toString());
- } catch (java.net.MalformedURLException me) {
- System.out.println("Malformed URL: " + me.toString());
- } catch (java.rmi.NotBoundException ne) {
- System.out.println("NotBound: " + ne.toString());
- }
- }
- }
五、編譯代碼:
要編譯 Java 源文件,請運行 javac 命令:
javac RmiSample.java RmiSampleImpl.java RmiSampleServer.java RmiSampleClient.java
六、爲遠程對象實現創建根和幹:
要創建存根程序和骨架文件,應以包含遠程對象實現的已編譯類包全名運行 rmic 編譯器。
存根(Stub)是遠程對象在客戶端的代理,它將RMI調用傳遞給服務器端的骨架(Skeleton),後者負責將該調用傳遞給實際的遠程方法輸入如下:
D:/RMI>rmic -d D:/RMI RmiSampleImpl 執行這個命令, 若rmic成功運行,RMI目錄下就會多出兩個新類: RmiSampleImpl_Stub.class RmiSampleImpl_Skel.class 它們分別對應的是存根(stub)和骨架(skeleton).
七、運行代碼:
運行服務端程序:在Windows下,輸入下列命令,在後臺啓動RmiSampleServer程序:
D:/RMI>java RmiSampleServer
運行客戶端程序:
D:/RMI>java RmiSampleClient
客戶端輸出: 1 + 2 = 3