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