最近做一个项目,需要在64位的VS2022 c++项目中调用32位的动态库,实现导出docx文件的功能,开始在网上找了一些解决方案,基本都是:创建32位COM组件 -- 注册32位COM组件 -- 64位程序调用32位COM组件,但是按照这些方法封装COM组件后,发现不能调用,客户端使用CoCreateInstance创建对象提示创建COM服务器接口的实例时出现类未注册错误(REGDB_E_CLASSNOTREG Class not registered,查看注册表,计算机\HKEY_LOCAL_MACHINE\SOFTWARE\WOW6432Node\Classes\CLSID\{**********},发现COM组件成功注册,但就是不能调用,百思不得其解!最后经过几天的摸索,找到解决办法,在这里分享一下。
一、COM组件有in-process和out-of-process两种
运行in-process的COM客户端和COM服务器需要以相同的位元运行,因此在x86中都是如此,或者在x64中都是这样。因此:
- 如果COM客户端以x86运行,它会与注册表的x86侧进行通信,因此COM服务器必须在注册表的x86侧注册(regsvr32、regasm、自定义reg代码等必须以x86方式运行)。
- 如果COM客户端以x64运行,它将与注册表的x64侧进行通信,因此COM服务器必须在注册表的x64侧注册(regsvr32、regasm、自定义reg代码等必须以x64方式运行)。
对于out-of-process通信,x86客户端或服务器可以与x64服务器或客户端进行通信。这是COM的一个好处。也是解决64位软件调用32位DLL的关键所在!笔者的问题就是封装了in-process的COM组件,导致无法在64位进程内调用32位DLL。后面创建out-of-process类型的COM组件,问题得到解决!
二、创建out-of-process类型的COM组件的流程
1.建一个ATL项目
2.选择可执行文件,注意:应用程序类型这里选择exe,表示创建out-of-process COM组件;如果选择DLL,则是in-process,切记不要选错
3.添加ATL简单对象
4.在IDL文件中添加 show([in] BSTR a);接口
import "oaidl.idl";
import "ocidl.idl";
[
object,
uuid(93e14b15-e9de-44c0-9fa1-12f5ba1265fd),
dual,
nonextensible,
pointer_default(unique)
]
interface IATLSimpleObject : IDispatch
{
[id(1)] HRESULT show([in] BSTR a);
};
[
uuid(74fa1274-a76d-4392-b2b1-f841db7be581),
version(1.0),
]
library ATLProject1Lib
{
importlib("stdole2.tlb");
[
uuid(d9049425-0435-4727-85bb-853d81368b19)
]
coclass ATLSimpleObject
{
[default] interface IATLSimpleObject;
};
};
import "shobjidl.idl";
5.在实现类中添加C++函数,实现在IDL新增的接口,此时可以调用x86的DLL,实现自己想要的功能
char* wchartoGBK(LPCWSTR wideStr) {
int wideStrLen = wcslen(wideStr);
int multiByteLen = WideCharToMultiByte(CP_ACP, 0, wideStr, wideStrLen, NULL, 0, NULL, NULL);
if (multiByteLen == 0) {
//std::cerr << "Error converting wide char to multi byte: " << GetLastError() << std::endl;
return "";
}
char* multiByteStr = new char[multiByteLen + 1];
if (!WideCharToMultiByte(CP_ACP, 0, wideStr, wideStrLen, multiByteStr, multiByteLen, NULL, NULL)) {
// std::cerr << "Error converting wide char to multi byte: " << GetLastError() << std::endl;
delete[] multiByteStr;
return "";
}
multiByteStr[multiByteLen] = '\0';
//delete[] multiByteStr;
return multiByteStr;
}
STDMETHODIMP_(HRESULT __stdcall) CATLSimpleObject::show(BSTR a)
{
auto str = wchartoGBK(a);
::MessageBoxA(NULL, str, "test", MB_OK);
delete[] str;
return S_OK;
}
6.编译选项选X86,编译,生成ATLProject1.exe文件
7.打开cmd窗口,cd生成目录,注册COM组件:ATLProject1.exe /regserver(反注册:ATLProject1.exe /unregserver),VS编译后会自动注册,次步可以省略
8.64位客户端调用:创建测试项目,Project1,编译选项选择x64,包含ATLProject1_i.h和ATLProject1_i.c文件,CoCreateInstance的第三个测试填CLSCTX_LOCAL_SERVER,运行弹出测试窗口,成功!收工下班(2024.7.26 18:29)
#include "../../ATLProject1/ATLProject1/ATLProject1_i.h"
#include <atlbase.h>
using namespace ATL;
int main() {
HRESULT hr_ini = ::CoInitialize(NULL);
IATLSimpleObject* Isimple = nullptr;
HRESULT hr1 = S_OK;
hr1 = ::CoCreateInstance(CLSID_ATLSimpleObject, NULL, CLSCTX_LOCAL_SERVER, IID_IATLSimpleObject, (LPVOID*)&Isimple);
if (SUCCEEDED(hr1)) {
BSTR bstr = ::SysAllocString(L"进程外调用!");
Isimple->show(bstr);
Isimple->Release();
}
::CoUninitialize();
}
输出结果:
参考连接: