圖1.1 OMG的CORBA參考模型 l Object Request Broker,ORB作為對(duì)象互通訊的軟總線。 l Object Services,定義加入ORB的系統(tǒng)級(jí)服務(wù),如安全性、命名和事務(wù)處理。 l Common Facilities定義應(yīng)用程序級(jí)服務(wù),如復(fù)合文檔等。 l application Interface 定義現(xiàn)實(shí)世界的對(duì)象和應(yīng)用,如飛機(jī)或銀行帳戶。 1.ObjectRequestBroker詳述 OMA最重要的部分就是ORB。為了創(chuàng)建一個(gè)遵從CORBA規(guī)范的應(yīng)用程序,ORB是CORBA四大部分中唯一必須提供的。許多ORB版本根本不帶CORBAServices或是CORBAFacilities,你可以自制(或購買)商用對(duì)象。但是,沒有ORB,CORBA應(yīng)用程序絕對(duì)無法工作。(圖5.2)
圖5-.2 單個(gè)ORB的體系結(jié)構(gòu) CORB ORB最顯見的功能,是對(duì)你的應(yīng)用程序或是其它ORB的請(qǐng)求予以響應(yīng)。在CORBA應(yīng)用程序運(yùn)行期間,你的ORB可能被請(qǐng)求做許多不同的事情,包括: l 查找并調(diào)用遠(yuǎn)程計(jì)算機(jī)上的對(duì)象 l 負(fù)責(zé)不同編程語言之間的參數(shù)轉(zhuǎn)換(如C++到Java) l 可超越本機(jī)界限的安全治理 l 為其它的ORB收集并發(fā)布本地對(duì)象的metadata l 用下載的代碼(stub)中描述的靜態(tài)方法調(diào)用去擊活遠(yuǎn)程對(duì)象中的方法 l 用動(dòng)態(tài)方法調(diào)用擊活遠(yuǎn)程對(duì)象 l 自動(dòng)擊活一個(gè)當(dāng)前沒有裝入內(nèi)存運(yùn)行的對(duì)象。 l 將回調(diào)方法導(dǎo)引向其治理之下的本地對(duì)象 實(shí)現(xiàn)細(xì)節(jié)對(duì)軟件開發(fā)者的透明性,是ORB的一個(gè)杰出的特性。用戶只須在代碼中提供相應(yīng)的hooks,用于初始化ORB并向ORB登記該應(yīng)用程序,就可以將該應(yīng)用程序和大量分布式對(duì)象建立聯(lián)系。 2.用IDL描述對(duì)象 為了保持CORBA的商業(yè)中立性和語言中立性,必須有一個(gè)中介,存在于象C++CORBA服務(wù)器代碼和JavaCORBA客戶機(jī)這樣的實(shí)體之間。這就是IDL。一個(gè)底層對(duì)象的若干相關(guān)方法和屬性被IDL集入一個(gè)單一接口。一旦IDL接口定義完成,它可以以stub碼或框架代碼(skeletoncode)的形式編譯成你選用的語言。在所有的ORB中都有IDL編譯器。例如,VisigenicVisiBrokerforJavaORB中就含有Java/IDL編譯器,而VisigenicVisiBrokerforC++ORB則提供了C++/IDL編譯器。 有一點(diǎn)值得注重的是IDL不同于其它的面向?qū)ο蟪绦蛟O(shè)計(jì)語言,我們不能用它指定它所定義的類或是方法的具體實(shí)現(xiàn)。因此,將它僅僅作為一種定義底層對(duì)象接口的語言要好得多。 就象在Java中將屬性和方法封裝到相關(guān)的類中一樣,上述各項(xiàng)均包含在IDL的模塊之中。在一個(gè)模塊之中可以定義一個(gè)或多個(gè)接口。表一中的簡(jiǎn)單IDL模塊名為TheModule,它含有一個(gè)稱為TheInterface的基本接口。該接口僅有一個(gè)定義為整型的簡(jiǎn)單變量(即TheVariable)。
2 用Java做CORBA開發(fā)實(shí)例 為了創(chuàng)建一個(gè)分布式的Java小應(yīng)用,并讓它用CORBA訪問服務(wù)器對(duì)象,我們利用一個(gè)流行的商用ORB,并用IDL定義對(duì)象接口。在示例小應(yīng)用中,我們選用了Visigenic VisiBroker for Java。這種ORB已經(jīng)經(jīng)過Oracle、Netscape和Novell等公司的認(rèn)證,并已被納入NetscapeNavigator4.0。 注重:你可以在非NetscapeNavigator4.0的瀏覽器中運(yùn)行這個(gè)小應(yīng)用。由于它首先要從別處下載一些Java類文件,啟動(dòng)速度可能會(huì)稍慢一些。 我們將用一個(gè)簡(jiǎn)單的Java小應(yīng)用調(diào)用一個(gè)使用CORBA的服務(wù)器對(duì)象。為簡(jiǎn)單起見,同樣用Java書寫服務(wù)器對(duì)象。該服務(wù)器對(duì)象用一個(gè)數(shù)組存儲(chǔ)有關(guān)各種CORBAORB開發(fā)商及他們產(chǎn)品的信息。客戶小應(yīng)用將調(diào)用該對(duì)象并查詢數(shù)組。一個(gè)更為完整的例子(仔細(xì)思考一下)是將ORB信息存儲(chǔ)于關(guān)系數(shù)據(jù)庫中,利用JDBC(或是別的數(shù)據(jù)庫訪問方法)獲得相關(guān)信息。這種方法將用CORBA生成一個(gè)真正的三層應(yīng)用程序。 1.最簡(jiǎn)單的IDL模塊 Module TheModule { interface TheInterface { long TheVariable; }; }; 假如你用一個(gè)IDL到Java的編譯器編譯這個(gè)IDL模塊(如Visigenic的idl2java),就會(huì)得到表1-2中的Java接口。 表1-2與TheModule相應(yīng)的Java代碼 package TheModule; public interface TheInterface { public int TheVariable; } 2.ORBQuery小應(yīng)用 這個(gè)客戶端的小應(yīng)用含有標(biāo)準(zhǔn)的JavaGUI,并將調(diào)用一個(gè)遠(yuǎn)程CORBA對(duì)象。一旦該對(duì)象被調(diào)用,就可以使用其方法獲得某一指定CORBA ORB的信息。在服務(wù)器端,為了獲得特定ORB的如下信息:名稱(Name)、銷售商(Vendor)、操作系統(tǒng)(OperatingSystem)、語言(Languages)和URL,我們必須定義五個(gè)方法。因此,必須在IDL接口中定義這五種方法才能獲取相應(yīng)信息。表1-3定義了這個(gè)名為ORBInfo的接口: 表1-3:ORBInfoIDL界面 module ORBQuery { interface ORBInfo { string GetName(in long index); string GetVendor(in long index); string GetOS(in long index); string GetLanguages(in long index); string GetURL(in long index); }; };VisiBroker安裝中含有一個(gè)IDL編譯器──idl2java,你可以用它生成實(shí)現(xiàn)該接口必需的Java代碼。軟件安裝完成之后,只要執(zhí)行如下命令即可生成代碼: idl2javaORBInfo.idl 這步操作將創(chuàng)建一個(gè)名為ORBQuery的子目錄(與ORBQueryJava包相對(duì)應(yīng))。在該目錄內(nèi)有8個(gè)文件:ORBInfo.java,ORBInfoHolder.java、ORBInfoHelper.java、_st_ORBInfo.java,_sk_ORBInfo.java、ORBInfoOperations.java、_tie_ORBInfo.java和_example_ORBInfo.java。你可能已經(jīng)猜到,ORBInfo.java文件含有定義ORBInfo接口的Java源文件,但其它的Java類又怎樣呢? ORBInfoHolder.java文件內(nèi)含有一個(gè)傳遞參數(shù)時(shí)使用的主類(holderclass),而ORBInfo-Helper類則定義了各種實(shí)用函數(shù)。_st_ORBInfo類定義了客戶stub,_sk_ORBInfo定義了服務(wù)器框架類(skeletonclass)。 ORBInfoOperations和_tie_ORBInfo類用于實(shí)現(xiàn)一種捆綁機(jī)制,這是VisiBroker的一個(gè)特性,它使得實(shí)現(xiàn)類能夠繼續(xù)框架類之外的類。在示例中,我們不會(huì)直接使用這幾個(gè)類。最后,_example_ORBInfo含有一個(gè)示例服務(wù)器對(duì)象,對(duì)它加以擴(kuò)展就可創(chuàng)建一個(gè)服務(wù)器應(yīng)用程序。 通過IDL編譯器生成的這八個(gè)Java類,我們可以構(gòu)建一個(gè)框架,由一個(gè)接口(interface)、一個(gè)stub、一個(gè)skeleton和幾個(gè)幫助類,我們可以用Java創(chuàng)建自己的客戶機(jī)/服務(wù)器模式的CORBA應(yīng)用程序。 3.創(chuàng)建服務(wù)器應(yīng)用程序 下面,我們需要?jiǎng)?chuàng)建一個(gè)向服務(wù)器ORB登記ORBInfo對(duì)象的服務(wù)器應(yīng)用程序。這個(gè)新對(duì)象將擴(kuò)充框架類(skeletonclass)并實(shí)現(xiàn)ORBInfo接口。因此,該服務(wù)器共需兩個(gè)新類:一個(gè)用于定義服務(wù)器對(duì)象并實(shí)現(xiàn)ORBInfo接口,另一個(gè)向服務(wù)器ORB登記該對(duì)象。ORBQuery類包含的標(biāo)準(zhǔn)Java代碼負(fù)責(zé)取回?cái)?shù)組中的指定元素。Server類中含有CORBA特有的功能。 接下來的例子中,開始是初始化ORB。然后,用“ORBInfo”字符串向ORB登記類,客戶機(jī)利用這個(gè)字符串檢索一個(gè)對(duì)象。所有的操作完成后,調(diào)用boa.obj_is_ready(),通知ORB一切預(yù)備就緒。 表1-4是Server類,它向ORB登記了ORBInfo對(duì)象。 表1-4 服務(wù)器類 public class Server { public static void main(String[] args) { try { // Initialize the ORB. org.omg.CORBA.ORB orb = org.omg.CORBA.ORB.init(); // Initialize the BOA. org.omg.CORBA.BOA boa = orb.BOA_init(); // Create the ORBQuery. ORBQuery serverQuery = new ORBQuery("ORBInfo"); // EXPort the newly create object. boa.obj_is_ready(serverQuery); System.out.PRintln(serverQuery + " is ready."); // Wait for incoming requests boa.impl_is_ready(); } catch(org.omg.CORBA.SystemException e) { System.err.println(e); } } }
public java.lang.String GetOS(int index) { String OS; OS = ORBVendors[index][2]; return OS; }
public java.lang.String GetLanguages(int index) { String Languages; Languages = ORBVendors[index][3]; return Languages; }
public java.lang.String GetURL(int index) { String URL; URL = ORBVendors[index][4]; return URL; } } 至此,我們已經(jīng)寫好了所有服務(wù)器方必需的代碼,下一步的工作是創(chuàng)建客戶小應(yīng)用,初始化客戶ORB,進(jìn)而擊活并調(diào)用剛剛生成的服務(wù)器對(duì)象。 4.創(chuàng)建CORBA小應(yīng)用 正如服務(wù)器對(duì)象要向服務(wù)器ORB登記一樣,客戶端小應(yīng)用或應(yīng)用程序需要向客戶機(jī)ORB登記。當(dāng)要獲得遠(yuǎn)程CORBA對(duì)象時(shí),客戶機(jī)采用了一種間接的方法,它通知客戶機(jī)ORB其意圖,由ORB負(fù)責(zé)ORB到ORB的通訊。這種請(qǐng)求方式由下面兩行代碼實(shí)現(xiàn)(在VisiBrokerforJava中): //初始化ORB(使用applet) org.omg.CORBA.ORBorb=org.omg.CORBA.ORB.init(this); //檢索applet要調(diào)用的ORBInfo接口對(duì)象 ORBInfoQuery=ORBQuery.ORBInfoHelper.bind(orb,"ORBInfo"); 執(zhí)行了bind()方法調(diào)用之后,我們的ORBInfoQuery本地變量與服務(wù)器的ORBInfo對(duì)象綁在一起。這一操作完成之后,我們就可以調(diào)用幫助方法來實(shí)現(xiàn)客戶端小應(yīng)用。記住,盡管示例中我們完全使用了Java語言,但實(shí)際上,服務(wù)器對(duì)象也可以用其他語言來實(shí)現(xiàn),如:COBOL、C++、Ada和Smalltalk。 5.編寫簡(jiǎn)單的CORBA服務(wù)的一般流程 (以Java2為例): l 編寫所需要的接口IDL文件。 n foo.idl module foo{ interface function{ float square_root(in float number); } } l 用idltojava編譯idl文件。 n idltojava ?fno-cpp foo.idl l 用Javac編譯所產(chǎn)生的類。 n javac ~*.java l 生成實(shí)現(xiàn)類。 n (functions.java) functinosImpl.java l 生成實(shí)現(xiàn)服務(wù)器。 n fooServer.java l 生成客戶機(jī)應(yīng)用程序(或小程序)。 n fooClient.java l 編譯實(shí)現(xiàn)服務(wù)器和客戶機(jī)代碼。 n javac functionsImpl.java fooServer.java fooClient.java l 啟動(dòng)命名服務(wù)應(yīng)用程序tnameserv。 n tnameserv ?ORBInitialPort 1080 l 啟動(dòng)服務(wù)器(用命名服務(wù)注冊(cè))。 n java fooServer ?ORBInitialPort 1080 l 啟動(dòng)客戶機(jī)。 n java fooClient ?ORBInitialPort 1080 6.結(jié)論 與單純的Java小應(yīng)用相比,創(chuàng)建CORBAJava應(yīng)用稍顯復(fù)雜。事實(shí)上,這同樣比用RMI創(chuàng)建純粹的Java分布式對(duì)象應(yīng)用要難一些。但是,CORBA和Java的聯(lián)合,所創(chuàng)建的應(yīng)用程序功能更為強(qiáng)大,更具可擴(kuò)充性,這是單純用JDK開發(fā)所無法比擬的。另外,現(xiàn)實(shí)是企業(yè)界很少使用Java應(yīng)用程序。CORBA是兩全其美的選擇,一方面它答應(yīng)開發(fā)者利用已有的代碼保護(hù)投資,另一方面又可以充分發(fā)揮Java編程語言的優(yōu)勢(shì)。