最近在做代码升级64位,因为有些第三方的模块没有源码,所以需要用到64位代码调用32位模块的方案,在网上找了很多资料和方法,最终参考国外的一篇文章解决了问题,原理是应用进程外com实现,具体见参考文献:
https://blog.mattmags.com/2007/06/30/accessing-32-bit-dlls-from-64-bit-code/。
现在为了更多的朋友少走弯路,这里把我实现的过程和源码跟大家分享。
步骤一:生成32位模块(如果已经有32位模块,这步骤可以跳过)
新建MFC DLL项目,项目名称为Add,新建接口文件AddExport.h然后添加接口方法AddMethod,实现两个整数的求和操作,声明如下:
int AddMethod(int num1, int num2);
编译生成32位的Add.dll模块。
步骤二:建立进程外Com,对32位模块进行封装
因为此步骤是关键步骤,所以写明具体操作步骤。
1. 新建ATL项目
项目名称为ComAddDemo,“确定”——》“下一步”,在弹出的界面中“应用类型”中选择第二项见下图红框圈出部分,然后“完成”。
- 添加接口类
切换到类视图,选中项目文件,右键——》添加——类,在弹出的界面左侧选择ATL,右面选择ATL简单对象,见下图。
点击“添加”,然后在弹出的向导界面中输入类名和ProgID,其他参数默认,一路下一步到完成。
注意:1)这里的ProgID一定要填,不然后面找不到入口将无法正确调用组件。
2)如果后面用到的时候忘记此处输入的名称,可以到“解决方案资源管理器”视图中的“资源文件”中的ISimAddObj.rgs文件中查看,内容如下:
添加完成后,在类视图的项目下会出现接口 。
3. 添加接口方法
选中上一步添加的接口,依次点击:右键——添加——方法,然后在向导界面中输入方法名并添加输入输入参数,见下图。这里输入参数勾选“in”;输出参数先选择LONG*类型,点亮“out”和“retval”,然后勾选“retval”。然后输入参数名后“添加”即可。
在“下一步”的界面中,可以在红框圈出的位置添加此方法的说明信息,然后点“完成”即可。
切换到解决方案资源管理器,展开“源文件”,双击打开ComAddDemo.idl文件,看到如下代码,就是刚才定义的COM方法。
方法实现
双击打开上面“源文件”中的ISimAddObj.cpp文件,调用步骤一中生成的Add.dll,编辑实现Add方法。如下图:编译运行
会生成32位的ComAddDemo.exe文件,并自动注册到系统中。
本步骤问题及解决:
1) 我在Win10 64位系统上编译的,出现下面错误信息
1>C:\Program Files (x86)\MSBuild\Microsoft.Cpp\v4.0\Microsoft.CppCommon.targets(113,5): error MSB3073: The command “”E:\VC Codes\VS2010\OOPComTest\ComAddDemo\Debug\ComAddDemo.exe” /RegServer
1>C:\Program Files (x86)\MSBuild\Microsoft.Cpp\v4.0\Microsoft.CppCommon.targets(113,5): error MSB3073: :VCEnd” exited with code -2147024891.
这是Windows权限问题,导致注册失败,退出VS2010,然后选中VS2010图标,右键选中“以管理员身份运行”,打开刚才的解决方案,重新编译ComAddDemo项目即可。
2) 用到的Add.dll需要和ComAddDemo.exe在同一目录下,才能注册成功。
步骤三:调用COM模块
1. 新建Win32控制台项目
项目名ComAddDemoTest,为了保证输出的文件在同一目录下,在项目向导界面中的解决方案选项如下,其他参数默认。
- 实现调用
在SDK中调用最简单最常用的一种,用#import导入类型库,利用VC提供的智能指针包装类。
在解决方案资源管理器视图中双击ComAddDemoTest.cpp文件,实现代码如下:
// ComAddDemoTest.cpp : Defines the entry point for the console application.
//
*#include “stdafx.h”
include
include
include
import “E:\VC Codes\VS2010\OOPComTest\ComAddDemo\ComAddDemo\Debug\ComAddDemo.tlb” raw_interfaces_only, raw_native_types, no_namespace, named_guids
in*t _t**main(int argc, _TCHAR* argv[])
{
long num1 = 10;
long num2 = 20;
long sum = 0;
CoInitialize(NULL);
CLSID clsid;
IISimAddObj* pProxy;
HRESULT hr = CLSIDFromProgID(OLESTR(“ComAddDemo.SimAddObj”), &clsid);//这里的字符串一定要和注册表中的ProgID一致!!!
hr = CoCreateInstance(clsid, NULL, CLSCTX_LOCAL_SERVER, IID_IISimAddObj, (void **)&pProxy); //创建COM对象
pProxy->Add(num1, num2, &sum); //调用COM接口
CoUninitialize();
printf("%d + %d = %d", num1, num2, sum);
return 0;
}
关于CLSIDFromPro*g*ID(OLESTR(“ComAddDemo.SimAddObj”), &clsid)中的字符串ComAddDemo.SimAddObj,跟步骤二-2中的ProgID有关,如果忘记请在ComAddDemo.rgs文件中找(直接在磁盘文件中打开或者参考步骤二-1中的注意事项)。rgs是组件注册的脚本文件,当你注册组件时,组件内部便是调用了这个文件。
运行结果
编译运行,结果如下:实现64位调用
在ComAddDemoTest项目中添加64位编译选项,重新编译运行,结果依然如上。至此,实现了64位代码调用32位模块的方案。至于参考文献中提到的安全性问题,现在还没有考虑,后面用到再学习。
完整代码可以在下面链接中下载:
https://download.csdn.net/download/yanmanm/10615236
本文详细介绍了如何在64位环境下使用VS2010创建COM组件封装32位DLL,并实现64位应用程序调用该32位模块的方法。步骤包括生成32位DLL、创建ATL项目、添加接口和方法、注册COM组件以及在64位Win32控制台项目中调用COM组件。
9976

被折叠的 条评论
为什么被折叠?



