Angelo Lee's Blog
This is my kingdom .If i don't fight for it ,who will ?

一、注册.ocx控件:

方法一:1、先把XXX.ocx控件放入c:\windows\system32 文件夹下

              2、在cmd命令窗口运行:regsvr32.exe  XXX.ocx

方法二:

使用RegSvr32.exe(在System目录下)注册,语法为:
    RegSvr32  ocxname.ocx 
如果注册成功将显示一个成功的消息框。
  
如果不希望看到这个对话框,也可以使用/s选项,如
    RegSvr32 /s your.ocx
如果取消注册,可以使用/u选项,如
    RegSvr32 /u your.ocx


二、取得控件的classid,开始-->运行-->输:regedit,打开注册表---->在注册表的工具栏的编辑中找到搜索按钮,输入XXX.ocx的名字XXX查找,就可以得到控件的classid

三、Java Invoke OCX Methods:

1、JNI

2、jacob  jacob official website

3、JNative Official Website



从官网上下载jacob.jar和jacob.dll,注意acob.jar和Jacob.dll需要统一版本。

  1. jacob.jar: a JAR file for the java classes which you must add to your CLASSPATH. The package names replace com.ms with com.jacob (for example com.ms.com.Variant maps tocom.jacob.com.Variant.
  2. jacob.dll: a small Win32 DLL which you must add to your PATH.

在一般的Win32中,可将Jacob.dll拷贝至C:\WINDOW\system32目录下,加入Jar包至Java项目中后如果仍存在问题,可将Jacob.dll拷贝至JVM的BIN目录下,例如D:\Java\jdk1.6.0_30\jre\bin



Java代码:

1.初始化

ActiveXComponent com = new ActiveXComponent("组件的ProgID") ;

Dispatch disp = com.getObject();

2.调用控件里面的方面

2.1调用无参的方法,并返回一个short值

Dispatch.call(disp, "Init").getShort();

2.2调用有一个参数的方法,并返回一个boolean值

Dispatch.call(disp,"Method",new Variant(args)).getBoolean();

调用多个参数依次类推,注意在传递参数前,将Java中的参数转换成Variant。

bug解决:

1.java.lang.UnsatisfiedLinkError: no jacob in java.library.path

原因是:没有把 jacob.dll 拷贝到 jdk\bin或者C:\WINDOWS\system32目录下

2.Exception in thread "main" com.jacob.com.ComFailException: Can't get object clsid from progid

原因是:regsvr32 ***.ocx命令进行注册后 ,获取的progid有误;

3.

Exception in thread "main" com.jacob.com.ComFailException: A COM exception has been encountered:

At Invoke of: LoadFile

Description: 灾难性故障

at com.jacob.com.Dispatch.invokev(Native Method)

at com.jacob.com.Dispatch.invokev(Dispatch.java:858)

at com.jacob.com.Dispatch.callN(Dispatch.java:455)

at com.jacob.com.Dispatch.call(Dispatch.java:544)

at com.lv.office.AIPToTest.jacob(AIPToTest.java:24)

at com.lv.office.AIPToTest.main(AIPToTest.java:14)

原因是:

在OCX控件中,重写COleControl.IsInvokeAllowed (DISPID)方法,即在控件的CMyNameCtrl中增加一个方法

BOOL CMyNameCtrl::IsInvokeAllowed (DISPID)

{

return TRUE;

}

用这个方法的理由是:

这个错误是由ActiveX结构设计造成的。

在Ole4.0版本之前,外部程序是可以直接调用OCX中方法的。Ole4.0之后,每次调用控件中的方法,系统会自动检查是否允许调用,即运行COleControl.IsInvokeAllowed (DISPID)

该方法检查控件是否正确的初始化或者是否通过持久存储接口正确加载,如果两个条件有一个满足,即返回TRUE,否则返回FALSE。

当控件在MFC中使用时,很多细节,如初始化,都被过滤了,这样,大多数用户都不会遇到这个问题。但是,当我们从C、C++的dll中调用控件时,不满足上述条件,该方法返回FALSE,这时候再调用任何控件方法,都会出现上述异常。


posted on 2012-11-06 13:08  Angelo Lee  阅读(795)  评论(1编辑  收藏  举报