淺論Java訪問(wèn)COM/ActiveX
發(fā)表時(shí)間:2024-02-21 來(lái)源:明輝站整理相關(guān)軟件相關(guān)文章人氣:
[摘要]Java作為一種跨平臺(tái)的語(yǔ)言,在很多環(huán)境下都獲得了成功。然而,在Windows平臺(tái)下,Java的發(fā)展卻受到了一定程度的限制。其中很重要的原因就是,目前Java對(duì)Windows構(gòu)件模型的支持力度不夠,使得Java程序很難復(fù)用Windows平臺(tái)下豐富的構(gòu)件資源,例如日歷、制表、Word等各種控件(CO...
Java作為一種跨平臺(tái)的語(yǔ)言,在很多環(huán)境下都獲得了成功。然而,在Windows平臺(tái)下,Java的發(fā)展卻受到了一定程度的限制。其中很重要的原因就是,目前Java對(duì)Windows構(gòu)件模型的支持力度不夠,使得Java程序很難復(fù)用Windows平臺(tái)下豐富的構(gòu)件資源,例如日歷、制表、Word等各種控件(COM/ActiveX)。
Windows構(gòu)件模型是基于COM的,目前JDK沒(méi)有提供任何直接訪問(wèn)COM的類庫(kù)。因此,如果需要訪問(wèn)這些資源,我們必須通過(guò)JNI實(shí)現(xiàn)。JNI是Java世界和其它語(yǔ)言間的一座橋,Java通過(guò)訪問(wèn)JNI定義的接口來(lái)獲取服務(wù)。在JNI的另一面,我們可以通過(guò)C/C++或其它語(yǔ)言實(shí)現(xiàn)這些接口。通過(guò)本地語(yǔ)言C/C++我們可以創(chuàng)建COM構(gòu)件,并且使用COM的服務(wù),最后將結(jié)果返回給Java程序。
在這里,我們涉及到幾個(gè)關(guān)鍵問(wèn)題。
1)數(shù)據(jù)類型的轉(zhuǎn)換。
Java和其它的語(yǔ)言定義的數(shù)據(jù)類型不盡相同,這使得我們需要對(duì)這些數(shù)據(jù)的進(jìn)行類型轉(zhuǎn)換。在Windows中,自動(dòng)化COM對(duì)象使用VARIANT作為其主要數(shù)據(jù)類型。VARIANT類型是對(duì)普通類型的一個(gè)封裝,我們很容易將它轉(zhuǎn)換成Java對(duì)應(yīng)得類型。例如,VARIANT中的VARIANT_BOOL可以直接對(duì)應(yīng)Java中的boolean。但是,一些其它數(shù)據(jù)類型的轉(zhuǎn)換看起來(lái)就比較麻煩,例如SAFEARRAY和一些指針。因此,在實(shí)現(xiàn)中通常在Java中定義一些Wrapper類型。
2)GUI處理
Windows下有大量ActiveX控件,都提供了界面服務(wù)。這些類的封裝性都非常好,具有很高的復(fù)用性。這些類實(shí)現(xiàn)了IDispatch接口,因此它們的使用也比較簡(jiǎn)單。但是,Java的窗口管理與Windows的窗口管理有很大差異。Windows利用句柄管理窗口。Java通過(guò)窗口類管理,對(duì)于重型構(gòu)件(AWT窗口),每一個(gè)構(gòu)件都有一個(gè)同位體,即存在一個(gè)本地窗口與之對(duì)應(yīng)。對(duì)于輕型構(gòu)件(Swing的大部分類),它們都沒(méi)有同位體。因此,我們可以考慮在重型構(gòu)件上放置ActiveX控件。
以下我們給出一個(gè)例子說(shuō)明,說(shuō)明如何使用同位體技術(shù),實(shí)現(xiàn)在Java的Panel上放置一個(gè)IE控件。
首先,在Java 程序中我們通過(guò)同位體的方法,獲一個(gè)Panel的同位體的窗口句柄。其中句柄用一個(gè)int表示。
public int getHWND()
{
int hwnd = 0;
DrawingSurfaceInfo drawingSurfaceInfo = ((DrawingSurface)(getPeer())).getDrawingSurfaceInfo(); //獲取同位體信息
if (null != drawingSurfaceInfo)
{
drawingSurfaceInfo.lock();
Win32DrawingSurface win32DrawingSurface = (Win32DrawingSurface)drawingSurfaceInfo.getSurface();
hwnd = win32DrawingSurface.getHWnd();//獲取同位體窗口句柄
drawingSurfaceInfo.unlock();
}
return hwnd;
}
然后,我們?cè)谕ㄟ^(guò)JNI方法,將這個(gè)句柄傳遞給C/C++程序。C/C++程序通過(guò)這個(gè)句柄創(chuàng)建ActiveX,這樣就可以實(shí)現(xiàn)將IE的ActiveX放在Java的Panel中。該例子使用ATL,并使用了相關(guān)的數(shù)據(jù)類型,如CComPtr等。
//產(chǎn)生IE控件
void CreateIEControl(ThreadParam *pThreadParam)
{
AtlAxWinInit();
// 第2個(gè)參數(shù)表示控件的ProgID或者 UUID,此例中使用IE控件。
HWND hwndChild = ::CreateWindow("AtlAxWin",
"Shell.Explorer.1",
WS_CHILD WS_VISIBLE,
0,0,0,0,
pThreadParam.hwnd,NULL,
//其中pThreadParam.hwnd就是在Java中獲取得據(jù)柄,作為父窗口。
::GetModuleHandle(NULL),
NULL);
IUnknown *pUnk = NULL;
AtlAxGetControl(hwndChild,&pUnk);
//讓IE訪問(wèn)pThreadParam.szURL所代表的URL
CComPtr spBrowser;
pUnk->QueryInterface(IID_IWebBrowser2, (void**)&spBrowser);
if (spBrowser)
{
CComVariant ve;
CComVariant vurl(pThreadParam.szURL);
spBrowser->put_Visible(VARIANT_TRUE);
spBrowser->Navigate2(&vurl, &ve, &ve, &ve, &ve);
}
}
3)事件通知
在COM中,外部事件通知是通過(guò)可連接對(duì)象實(shí)現(xiàn)的,客戶程序通過(guò)訪問(wèn)COM組件的出接口,以實(shí)現(xiàn)登記一個(gè)事件的接收器。這種事件通知模式和Java的事件代理模式非常類似。因此,如果要在Java中實(shí)現(xiàn)COM的事件通知,就要在Java程序中實(shí)現(xiàn)自定義事件監(jiān)聽(tīng)類,并將COM的事件接收器登記在Java程序中。這樣,COM的事件就可以通知到Java程序。
本文只是淺析了Java訪問(wèn)COM的基本原理,在實(shí)際應(yīng)用中,雖然可能有不同的解決方案,但基本原理都上文所述。另外,一些機(jī)構(gòu)和個(gè)人提供了一些Java和COM的軟件包,使得這種訪問(wèn)更加方便。例如,JavaCom 、Jacob和IBM提供的eclipse軟件包等等。