浅论Java访问COM/ActiveX

  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软件包等等。