因为一些原因,我们写64位程序时需要调用32位的dll,或者32位程序需要调用64位的dll
项目中我也遇到了这么一个没法回避的问题
看网上有写什么进程外com,没去研究,我用了一个比较简单的方案可以实现
主要就两个方案的结合
1.WM_COPYDATA
2.共享内存
当a程序SendMessage发送WM_COPYDATA 时,b程序执行WM_COPYDATA处理函数,此时a程序阻塞在SendMessage处,只有b程序执行处理函数结束后,a程序才继续执行SendMessage后的语句。这样就可以实现在A程序需要调用dll函数时,通过发送WM_COPYDATA消息,让b程序去调用需要的dll,然后将结果返回到a程序。
WM_COPYDATA消息是传输少量只读数据给其他程序的,所以可以把调用dll所需的参数值传递给b程序。b程序调用dll后如果将结果给a程序呢?这里就可以用共享内存。
b程序开一个共享内存,将结果写入共享内存,当a程序等到b程序 执行完WM_COPYDATA后,通过共享内存就拿到dll函数执行结果。
a程序可以在启动时打开b程序,退出时关闭b程序。b程序在启动时就开一个共享内存(当然,如果dll函数执行结果长度实在没办法用一个足够的缓冲区定义大小,也可以每次执行dll函数时开一个共享内存),并且b程序ShowWindow(SW_HIDE)隐藏掉。
实际的代码流程假设这样
struct PT
{
PT() :x(-1), y(-1) {}
int x;
int y;
}; 存储dll函数返回结果的结构体
=======================
一段代码 。。。。
int param = 10;
PT pt;
fun(param, &pt.x, &pt.y ); 这里调用了一个dll中的函数,这个函数进行了很复杂的运算出来个结果
int y = pt.y* 5...
一段代码。。。。。
=======================
a、b程序公用的定义
#define mapFileName _T("share") //共享内存名称
const int ShareSize = sizeof(PT);
#define PrjName _T("prj") //b程序窗口名。。由于b程序需要隐藏,在b程序初始化时SetWindowText(PrjName);
PT *m_pt;//b程序程序变量,指向共享内存中数据
=====第一步
b程序启动时开启共享内存
HANDLE hMapFile = CreateFileMapping(INVALID_HANDLE_VALUE, NULL, PAGE_READWRITE, 0, ShareSize, mapFileName);
m_pt = (PT*)MapViewOfFile(hMapFile, FILE_MAP_ALL_ACCESS, 0, 0, ShareSize);//
=====第二步
a程序执行函数时发送WM_COPYDATA消息
int param = 10;
HWND hWnd = ::FindWindow(NULL, PrjName);
if (hWnd == NULL) { return; }
COPYDATASTRUCT cds;
cds.dwData = 0;
cds.lpData = ¶m;//可以是结构体
cds.cbData = sizeof(param);
SendMessage(hWnd, (UINT)WM_COPYDATA, (WPARAM)m_hWnd, (LPARAM)&cds);
======第三步
b程序处理WM_COPYDATA消息
BOOL C**Dlg::OnCopyData(CWnd* pWnd, COPYDATASTRUCT* pCopyDataStruct)
{
int param = (int)pCopyDataStruct->lpData;
fun(param, &m_pt->x,& m_pt->y); //dll中的函数,m_pt已经是共享内存指针,直接进行操作写入共享内存
return CDialog::OnCopyData(pWnd,pCopyDataStruct);
}
======第四步
a程序从共享内存拿到结果
紧跟着SendMessage(hWnd, (UINT)WM_COPYDATA, (WPARAM)m_hWnd, (LPARAM)&cds);这句之后
HANDLE hMapFile = OpenFileMapping(FILE_MAP_ALL_ACCESS, FALSE, mapFileName);
if (hMapFile == NULL) return NULL;
PT *ptr = (PT*)MapViewOfFile(hMapFile, FILE_MAP_ALL_ACCESS, 0, 0, ShareSize);
CloseHandle(hMapFile);/// 一定要关句柄。。。。
int y = ptr->y* 5... 对结果进行使用
经过这个流程下来就实现了调用与程序位数不同的dll
当然这个是单线程的,多线程就得开不同共享内存