[摘要]Java作为一种跨平台的语言,在很多环境下都获得了成功。然而,在Windows平台下,Java的发展却受到了一定程度的限制。其中很重要的原因就是,目前Java对Windows构件模型的支持力度不够,...
Java作为一种跨平台的语言,在很多环境下都获得了成功。然而,在Windows平台下,Java的发展却受到了一定程度的限制。其中很重要的原因就是,目前Java对Windows构件模型的支持力度不够,使得Java程序很难复用Windows平台下丰富的构件资源,例如日历、制表、Word等各种控件(COM/ActiveX)。
Windows构件模型是基于COM的,目前JDK没有提供任何直接访问COM的类库。因此,如果需要访问这些资源,我们必须通过JNI实现。JNI是Java世界和其它语言间的一座桥,Java通过访问JNI定义的接口来获取服务。在JNI的另一面,我们可以通过C/C++或其它语言实现这些接口。通过本地语言C/C++我们可以创建COM构件,并且使用COM的服务,最后将结果返回给Java程序。
在这里,我们涉及到几个关键问题。
1)数据类型的转换。
Java和其它的语言定义的数据类型不尽相同,这使得我们需要对这些数据的进行类型转换。在Windows中,自动化COM对象使用VARIANT作为其主要数据类型。VARIANT类型是对普通类型的一个封装,我们很容易将它转换成Java对应得类型。例如,VARIANT中的VARIANT_BOOL可以直接对应Java中的boolean。但是,一些其它数据类型的转换看起来就比较麻烦,例如SAFEARRAY和一些指针。因此,在实现中通常在Java中定义一些Wrapper类型。
2)GUI处理
Windows下有大量ActiveX控件,都提供了界面服务。这些类的封装性都非常好,具有很高的复用性。这些类实现了IDispatch接口,因此它们的使用也比较简单。但是,Java的窗口管理与Windows的窗口管理有很大差异。Windows利用句柄管理窗口。Java通过窗口类管理,对于重型构件(AWT窗口),每一个构件都有一个同位体,即存在一个本地窗口与之对应。对于轻型构件(Swing的大部分类),它们都没有同位体。因此,我们可以考虑在重型构件上放置ActiveX控件。
以下我们给出一个例子说明,说明如何使用同位体技术,实现在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++程序通过这个句柄创建ActiveX,这样就可以实现将IE的ActiveX放在Java的Panel中。该例子使用ATL,并使用了相关的数据类型,如CComPtr等。
//产生IE控件
void CreateIEControl(ThreadParam *pThreadParam)
{
AtlAxWinInit();
// 第2个参数表示控件的ProgID或者 UUID,此例中使用IE控件。
HWND hwndChild = ::CreateWindow("AtlAxWin",
"Shell.Explorer.1",
WS_CHILD
关键词:浅论Java访问COM/ActiveX