使用进程外COM服务在64位应用中调用32位DLL
创始人
2024-10-22 06:11:40
0

最近做一个项目,需要在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   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(); }

输出结果:

参考连接:

VS2022使用进程外Com服务让64位应用调用32位DLL

裸写一个进程外 COM 组件

创建COM服务器接口的实例时出现类未注册错误

相关内容

热门资讯

wepoke辅助技巧(透视)w... wepoke辅助技巧(透视)wepower有机器人吗(详细辅助力荐教程)总是是有挂(教你辅助器)1、...
微扑克德州专用辅助器(微扑克)... 微扑克德州专用辅助器(微扑克)微扑克这软件有问题吗(透视)确实真的是有挂(详细辅助切实教程)1、这是...
wepoke真的有挂(透视)w... wepoke真的有挂(透视)wepoke有挂吗(详细辅助必备教程)竟然存在有挂(攻略ai机器人)we...
微扑克辅助器ios(微扑克)微... 微扑克辅助器ios(微扑克)微扑克怎么在软件内设置(透视)一直真的是有挂(详细辅助安装教程)小薇(透...
wepok软件透明挂(透视)w... wepok软件透明挂(透视)wepoke系统(详细辅助玩家教你)本来真的有挂(大神黑科技);wepo...
微扑克有辅助挂(微扑克)微扑克... 微扑克有辅助挂(微扑克)微扑克辅助软件(透视)切实有挂(详细辅助教你攻略)1、玩家可以在微扑克辅助软...
WePoKe外 挂(透视)we... WePoKe外 挂(透视)wepoke有挂吗网上(详细辅助教你教程)都是有挂(黑科技有外 挂)1、玩...
微扑克游戏辅助器(微扑克)微扑... 微扑克游戏辅助器(微扑克)微扑克智能助手(透视)一直是真的有挂(详细辅助玩家教你)1、游戏颠覆性的策...
wepower有外 挂(透视)... wepower有外 挂(透视)wepoke是不是有辅助(详细辅助扑克教程)一直存在有挂(玩家一定有挂...
微扑克wpk透视辅助(微扑克)... 微扑克wpk透视辅助(微扑克)微扑克辅助器ios(透视)一直有挂(详细辅助wpk教程)1、用户打开应...