一、先上伪代码
#include <QAxObject>
#include <qt_windows.h>
bool Init(char cMode)
{
QAxObject m_QAxObj;
QVariantList params0;
QString qresult;
int engine_hand;
HRESULT hres = CoInitializeEx(0, COINIT_MULTITHREADED );
if (hres != S_OK && hres != 0x80010106)
{
hres = CoInitializeEx( 0, COINIT_APARTMENTTHREADED );
if (hres != S_OK && hres != 0x80010106)
{
qWarning("Qt:初始化Ole失败(error %x)",(unsigned int)hres);
return 0;
}
}
b_re = m_QAxObj.setControl(QString::fromLocal8Bit("{7B3333C0-FDA0-11CE-8B06-444553540000}") );
params0.clear();
params0<<QVariant( 0 );
params0<<QVariant( qresult );
//int getHandle(int engin_type, QString &result)
engine_hand = m_QAxObj.dynamicCall( "getHandle(QVariant)", params0 );
}
首先包含COM组件需要使用的两个头件。 CoInitializeEx()函数返回的不一定是S_OK,因为如果在主进程中使用的话,组件初始化用QT预先调用了,这时返回值是0x80010106,因此此两种情况下都应该认识初始化成功。
调用COM服务器的函数时,使用dynamicCall函数传参数有两种方法,一种是直接写入参数,一种通过QVariantList 传入参数,如果参数有通过引用方法返回数据的时候,一定要用QVariantList方法。否则不能正确返回值。同时要注意如上面的“getHandle”函数返回结果字符串result不是直接放到QString qresult变量中,而是放到QVariantList params0变量中的。
二、组件导出函数说明
QT提供了打印COM组件函数说明方法,很强大,很好用
QString DOC = m_QAxObj.generateDocumentation();
QFile outFile("d:/cmcengine.html");
outFile.open(QIODevice ::ReadWrite | QIODevice ::Text);
QTextStream TS(&outFile);
TS<<DOC<<endl;
三、问题
1、一开始使用的时候遇个问题,就是第一次调用很好,第二次调用时就会出错报:
中的 0x7c812afb 处最可能的异常: 0x000006BA: RPC 服务器不可用
查了很久才终于发现,我的COM组件是在线程中调用的。第一次调用完就会结束线程,第二次调用重新开启线程,这就导致RPC服务器不可用。如果第一次调用完不关闭线程,第二次调用直接使用刚才的线程继续执行就不会出问题了。
2、在使用QT的过程中,使用QFileDialog::getOpenFileName()函数时,也会报KernelBase 第一机会处异常。但是程序没有任何问题,正常执行。我使用的是VS2012和64位的win7系统。后来换成VS2015一样错误。重装系统,重装vs都不能解决。百度说可以通过取消vs调试菜单中的异常--C++ EXceptions报警来取消这个报警,这个方法明显是取消报警,但是没有解决问题。我设置了一下,并没有启作用。仍然有这个报警。
后来发现在另外一台win64位机器上,不会报这个警,但是并没有发现这两台机器有什么不同。最终比较了一下这两台机器的windows/system32文件夹下的文件,发现有问题的这台机器的kernelbase.dll文件创建时间要早很多。没有问题的这台机器的KernelBase.dll文件是2019年的。偿试着把新的这个KernelBase.dll文件复制到有问题的这台机器上,因为有太多程序加载这个文件了,没办法复制。这时不能不装个某安全卫生,偿试修复系统,在升级了一堆windows补丁之后问题解决。