这个问题捣鼓了两天,现在终于解决了,做个笔记分享给大家,以免走弯路
起初,我的想法是在DLL中写一个interface并从函数中导出这个interface,像这样的代码
- ICom1 = interface
- function Show(V1, V2: Integer): Integer stdcall;
- end;
最后均以失败告终,后来想到各种编译器对编译后的二进制组织方式是不同的
比如上面的代码如果用Delphi编写的Exe去调用就是没问题的,而用其他语言则可能会有问题
网上有很多跨语言调用方案是用虚类来解决,也许有些时候是可以正常调用,但是这种做法并不规范,容易出问题
这里就需要通过一个标准,一个与语言无关的标准,也就是微软的COM组件了
所以得改下这个DLL
新建一个Automation Object
注意,这里一定要用Automation Object,而不要用COM Object,因为后者不支持自动化调用
而QAxObject只支持自动化对象,如果你尝试去调用一个不支持自动化的COM Object会得到一条消息
“QAxBase::dynamicCallHelper: Object does not support automation”
创建成功后接口是这样的
- ICOM2 = interface(IDispatch)
- ['{53952FF2-94A4-4B14-9C38-E4E56C87940A}']
- function Show(v1: Integer; v2: Integer): Integer; stdcall;
- end;
注意,我们在Exe中查询需要用到的GUID是类ID而不是这里的接口ID,在Delphi自动生成的 XXX_TLB.pas 文件中是有这个GUID的
我这里的是 {0EA6D9F4-0587-4AB9-91AD-9CD657B0787D}
最后实现
- function TCOM2.Show(v1, v2: Integer): Integer;
- begin
- OutputDebugString(PChar(Format('TCom2(%d, %d)', [v1, v2])));
- Result := 2;
- end;
现在到Qt中,首先写一个函数,功能是从一个COM DLL中动态创建接口实例
这样用的好处是COM DLL不用注册就能用,当然注册的话调用起来会更方便
- LPUNKNOWN CreateComObjectFromDll(const QString &dll, REFCLSID clsid)
- {
- QLibrary lib(dll);
- if (lib.load()) {
- typedef HRESULT (__stdcall *DllGetClassObject)(REFCLSID, REFIID, LPVOID*);
- DllGetClassObject getClassObject = (DllGetClassObject)lib.resolve("DllGetClassObject");
- if (getClassObject != nullptr) {
- IClassFactory *factory;
- if (getClassObject(clsid, IID_IClassFactory, (LPVOID*)&factory) == S_OK) {
- LPUNKNOWN ret = nullptr;
- factory->CreateInstance(nullptr, IID_IUnknown, (void**)&ret);
- return ret;
- }
- }
- }
- return nullptr;
- }
最后,调用部分
- LPUNKNOWN obj = CreateComObjectFromDll("com", cid);
- QAxObject o(obj);
- o.dynamicCall("Show(int, int)", 123, 456);
参数cid就是接口类GUID
可以看到输出窗口正常显示了结果。
总结:起初没有使用标准的COM接口,走了不少弯路。。。 -_-!
http://blog.csdn.net/aqtata/article/details/9163689