■CORBA技術簡介
簡單地說,CORBA允許應用之間相互通信,而不管它們存在於哪裏以及是誰設計的。CORBA1.1於1991年由OMG發佈,其中定義了接口定義語言(IDL)以及在對象請求代理(ORB)中實現客戶對象與服務器對象之間交互的應用編程接口(API)。CORBA2.0於1994年發佈,規定了各個供應商之間的ORB的通信規則。
CORBA標準主要分爲三個部分:接口定義語言(IDL)、對象請求代理(ORB)以及ORB之間的互操作協議IIOP。
ORB是對象之間建立Client/Server關係的中間件。使用ORB,客戶可以透明地調用一個服務對象上的方法,這個服務對象可以在本地,也可以在通過網絡連接的其他機器上。ORB截獲這一調用同時負責查找實現服務的對象並向其傳遞參數、調用方法返回最終結果。客戶並不知道服務對象位於什麼地方,它的編程語言和操作系統是什麼,也不知道不屬於對象接口的其他系統部分。這樣,ORB在異構分佈環境下爲不同機器上的應用提供了互操作性,並無縫地集成了多種對象系統。
在開發傳統的Client/Server應用時,開發者使用他們自己設計的或一個公認的標準來定義用於設備之間通信的協議。協議的定義依賴於實現語言、網絡傳輸和許多其他因素,而ORB的出現簡化了這一過程。使用ORB時,協議是使用接口定義語言(IDL)定義的,而IDL是獨立於語言的。並且ORB提供很強的靈活性,它使程序員選擇最適合的操作系統、執行環境,甚至系統各個組件也可以採用不同的編程語言實現。更重要的是,它允許現有組件的集成。在一個基於ORB的解決方案中,開發者可以使用與創建新對象一樣的IDL對遺留系統進行建模,他們創建“包裝”代碼以在標準化的軟件總線與遺留系統接口之間傳遞信息。
使用CORBA,用戶可以透明地訪問信息,並不需要知道信息存在於什麼軟件中、使用什麼硬件平臺,以及位於企業網絡的什麼地方。作爲面向對象系統的通信核心,CORBA爲今天的計算環境帶來了真正的互操作性。
■CORBA與JAVA的相互關係
CORBA不只意味着對象請求代理(ORB),它還是非常全面的分佈式對象平臺。CORBA使JAVA應用可以跨越網絡、語言以及操作系統,併爲JAVA提供了一組分佈服務,如分佈式自我觀察、動態發現、事務、關係、安全和命名等。
JAVA不僅是一種語言,它還是一個動態代碼系統,它對運行對象來說是一個可移植的虛擬機(JVM)。JAVA爲開發、管理、發佈Client/Server應用提供了更簡單的方式。人們可以通過將應用放在一個Web服務器上將這一應用發佈給成千上萬個用戶,而不必關心它的安裝和升級。JAVA還非常適合服務器的開發,它可以動態地將服務代碼移向最需要它們的地方。
JAVA將會使CORBA對象能夠運行在從主機、網絡計算機到蜂窩電話等可編程的各種機器上,並簡化了大型CORBA系統的代碼發佈。對客戶和服務對象來說JAVA是很理想的編程語言,JAVA內置的多線程、垃圾收集和錯誤處理使編寫健壯的網絡對象變得很容易。
這兩種對象模型可以很好地相互補充,CORBA處理網絡的透明性,JAVA處理實現的透明性,CORBA爲JAVA可移植應用環境提供了一個分佈式的結構。
■使用JAVA開發CORBA應用
下面讓我簡要介紹一下開發CORBA的步驟。
使用JAVA開發CORBA應用需要如下五個步驟:
1.使用IDL創建接口 (About.idl)
下面的OMG IDL描述一個CORBA對象。
module About
{
interface Show
{
string ShowName();
};
};
將其存爲Show.idl。
2.編譯接口並生成CORBA支持文件
我們用以下命令編譯這個 IDL 接口:
idltojava Show.idl
idltojava是SUN公司的IDL編譯器,可以免費從SUN公司站點上下載。
因爲idltojava在編譯IDL文件之前,需要進行預編譯,而如果你的機器上沒有預編譯器,可以使用以下命令:
idltojava -fno-cpp Show.idl
編譯後將在當前目錄下生成About子目錄,其中會包括一些支持文件,如有興趣可以看一下,但一定不要修改。
3.實現服務器 (ShowServer.java)
ShowServer的main() 方法,可完成以下任務:
(1)創建一個 ORB 實例
(2)創建一個服務對象實例(CORBA About 對象的實現)並通知 ORB
(3)獲取一個命名上下文的CORBA對象引用,在該命名上下文中註冊新的CORBA對象
(4)在命名上下文中將新對象註冊在“About”名下
(5)等待對新對象的調用
實現服務器源程序如下:
import About.;
import org.omg.CosNaming.;
import org.omg.CosNaming.NamingContextPackage.;
import org.omg.CORBA.;
class ShowObject extends _ShowImplBase
{
public String ShowName()
{
return "/nMy name is Seymour!!/n";
}
}
public class ShowServer {
public static void main(String args[])
{
try{
// 創建和初始化 ORB
ORB orb = ORB.init(args, null);
// 創建服務對象並將其向 ORB 註冊
ShowObject ShowRef = new ShowObject();
orb.connect(ShowRef);
// 獲取根命名上下文
org.omg.CORBA.Object objRef = orb.resolve_initial_references("NameService");
NamingContext ncRef = NamingContextHelper.narrow(objRef);
// 綁定命名中的對象引用
NameComponent nc = new NameComponent("About", "");
NameComponent path[] = {nc};
ncRef.rebind(path, ShowRef);
// 等待來自客戶機的調用
java.lang.Object sync = new java.lang.Object();
synchronized (sync) {
sync.wait();
}
} catch (Exception e) {
System.err.println("ERROR: " + e);
e.printStackTrace(System.out);
}
}
}
4.實現客戶機 (ShowClient.java)
下面的應用程序客戶機將完成以下任務:
(1)創建一個ORB;
(2)獲取一個指向命名上下文的引用;
(3)在命名上下文中查找“Show”並獲得指向該 CORBA 對象的引用;
(4)調用對象的 ShowName() 操作並打印結果。
import About.;
import org.omg.CosNaming.;
import org.omg.CORBA.;
public class ShowClient
{
public static void main(String args[])
{
try{
// 創建和初始化 ORB
ORB orb = ORB.init(args, null);
// 獲取根命名上下文
org.omg.CORBA.Object objRef = orb.resolve_initial_references("NameService");
NamingContext ncRef = NamingContextHelper.narrow(objRef);
//解析命名中的對象引用
NameComponent nc = new NameComponent("About", "");
NameComponent path[] = {nc};
About.Show ShowRef = ShowHelper.narrow(ncRef.resolve(path));
// 調用 Show 服務對象並打印結果
String show = ShowRef.ShowName();
System.out.println(show);
} catch (Exception e) {
System.out.println("ERROR : " + e) ;
e.printStackTrace(System.out);
}
}
}
5.構建和運行ShowName程序
(1)編譯 .java 文件,包括 stub 和 skeleton(在About目錄中):
javac .java About/.java
(2)啓動一個MS-DOS命令解釋器,輸入以下命令,確保名字服務器處於運行狀態:
tnameserv -ORBInitialPort 1050
(3)啓動另一個MS-Dos命令解釋器,輸入以下命令,啓動Show服務器:
java ShowServer -ORBInitialPort 1050
(4)再啓動一個MS-Dos命令解釋器Show應用程序客戶機:
java ShowClient -ORBInitialPort 1050
這時屏幕上會出現“My name is Seymour!”的字樣,說明實驗已經成功了。
本示例程序在Windows 98、Java 1.2.2平臺下實現。