簡析Java RMI (類似.net remoting)打造自己的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的定義
import java.rmi.*;
public interface RmiSample extends Remote {
public int sum(int a,int b) throws RemoteException;
}
二、實現遠程接口:
遠程對象實現類必須擴展遠程對象java.rmi.UnicastRemoteObject類,並實現所定義的遠程接口。遠程對象的實現類中包含實現每個遠程接口所指定的遠程方法的代碼。這個類也可以含有附加的方法,但客戶只能使用遠程接口中的方法。因爲客戶是指向接口的一個句柄,而不是它的哪個類。必須爲遠程對象定義構造函數,即使只准備定義一個默認構造函數,用它調用基礎類構造函數。因爲基礎類構造函數可能會拋出 java.rmi.RemoteException,所以即使別無它用必須拋出java.rmi.RemoteException例外。
以下是遠程對象實現類的聲明:
import java.rmi.*;
import java.rmi.server.*;
public class RmiSampleImpl extends UnicastRemoteObject
implements RmiSample {
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);
以下是服務器類的聲明:
import java.rmi.*;
import java.rmi.registry.*;
public class RmiSampleServer{
public static void main(String args[]) {
try {
LocateRegistry.createRegistry(8808) ;
SampleServerImpl Server = new SampleServerImpl();
// 將該對象實例與名稱“SAMPLE-SERVER”捆綁
Naming.rebind("//localhost:8808/SAMPLE-SERVER" , Server);
} catch (java.net.MalformedURLException me) {
System.out.println("Malformed URL: " + me.toString());
} catch (RemoteException re) {
System.out.println("Remote exception: " + re.toString());
}
}
}
四、編寫使用遠程服務的客戶機類:
客戶機類的主要功能有兩個,一是通過Naming.lookup方法來構造註冊服務程序 stub 程序實例,二是調用服務器遠程對象上的遠程方法。
以下是服務器類的聲明:
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());
}
}
}
五、編譯代碼:
要編譯 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