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|ws_visible,
0,0,0,0,
pthreadparam.hwnd,null,
//其中pthreadparam.hwnd就是在java中获取得据柄,作为父窗口。
::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中,外部事件通知是通过可连接对象实现的,客户程序通过访问com组件的出接口,以实现登记一个事件的接收器。这种事件通知模式和java的事件代理模式非常类似。因此,如果要在java中实现com的事件通知,就要在java程序中实现自定义事件监听类,并将com的事件接收器登记在java程序中。这样,com的事件就可以通知到java程序。
本文只是浅析了java访问com的基本原理,在实际应用中,虽然可能有不同的解决方案,但基本原理都上文所述。另外,一些机构和个人提供了一些java和com的软件包,使得这种访问更加方便。例如,javacom 、jacob和ibm提供的eclipse软件包等等。
闽公网安备 35060202000074号