Windows进程通信 -- WM_COPYDATA 消息
来自:https://blog.csdn.net/u012104827/article/details/102914600
windows 进程间通信,可以通过:SendMessage() -> WM_COPYDATA
关于 WM_COPYDATA 消息说明:https://docs.microsoft.com/zh-cn/windows/win32/dataxchg/wm-copydata
COPYDATASTRUCT有三个成员变量,如下所示:
typedef struct tagCOPYDATASTRUCT { ULONG_PTR dwData; DWORD cbData; PVOID lpData; } COPYDATASTRUCT, *PCOPYDATASTRUCT;
注意:
1、dwData为自定义数据,按照自己习惯设置就好,不影响对象的传输;
2、cbData,MSDN解释为:The size, in bytes, of the data pointed to by the lpData member.即lpData指向的数据的长度,要是这个变量的值设置错误,就会导致WM_COPYDATA传输数据失败;
3、lpData,传输的数据。使用简单的数据最好,例如char数组。本人在程序中使用了string对象,发现在子进程不能接收到正常的数据,使用char数组却很正常。有可能跟string对象不能跨进程访问有关,读者如果知道原因的话请在评论区告诉我,谢谢;
4、使用WM_COPYDATA时要用SendMessage而不能使用PostMessage,因为SendMessage是阻塞的,会等待消息响应窗体处理消息完毕后再返回;而PostMessage是异步的,这样就可能会导致当消息响应窗体接收到WM_COPYDATA的时候,COPYDATASTRUCT对象已经被析构了,导致访问数据发生异常;
5、由于使用SendMessage,所以不应该在WM_COPYDATA中处理数据,可以在消息响应窗体的WM_COPYDATA中先把COPYDATASTRUCT对象中的数据复制出来,通过自定义消息发送到消息响应窗体,然后立即返回,来减少父进程的阻塞时间。这样就把处理数据的代码放在自定义消息处了。
在另一篇文章中 https://blog.csdn.net/syb1295306116/article/details/104155986 ,有这么一段:
这里注意要分配全局内存,否则另一进程接受消息时,消息被释放,接受的是乱码,无法解析。
const UINT messageID = RegisterWindowMessage("SingletonApplication"); char szTemp[1024] = { 0 }; sprintf(szTemp, "LayeredWindow_%d", iWndID);//窗口名 HWND hTemp = ::FindWindow(NULL, szTemp);//窗口句柄 BYTE* pGlobal = (BYTE*)::GlobalAlloc(GMEM_FIXED, str.length());//全局内存 if (!pGlobal) { return; } else { ZeroMemory(pGlobal, str.length()); memcpy(pGlobal, str.c_str(), str.length()); } COPYDATASTRUCT copyData = {0}; copyData.dwData = messageID; copyData.cbData = (DWORD)(str.length()+1); copyData.lpData = pGlobal; SendMessage(hTemp, WM_COPYDATA, 0, (LPARAM)©Data); ::GlobalFree((HGLOBAL)pGlobal);//释放全局内存
不过,我测试的时候,使用 malloc() 没使用 GlobalAlloc() ,倒也没问题,不知何故?
-----------------------------------------------------------------------------------------------
我在一篇文章中,找到这么一段:https://bbs.csdn.net/topics/10798 (2000-07-06)
GlobalAlloc在Win16中就已经有了,这个函数返回一个句柄,通过这个句柄,两个进程可以共享一块内存,DDE和剪贴板就是通过这个函数交换数据的;malloc是一个库函数,这个函数的功能,是通过内部调用了VirtualAlloc完成的,并且分配的内存不能共享。
一般来说,涉及到DDE和剪贴版的内存操作使用GlobalAlloc,大多数程序内部的内存操作,使用malloc
如果要使用大块的共享内存,建议使用File Mapping
但是又在一篇文章中,看到:https://blog.csdn.net/wenzhou1219/article/details/17693241 (2014-01-02)
16位windows用一个全局堆和局部堆来管理内存,每一个应用程序或dll装入内存时,代码段被装入全局堆,而系统又为每个实例从全局堆中分配了一个64kb的数据段作为该实例的局部堆,用来存放应用程序的堆栈和所有全局或静态变量。而LocalAlloc/GlobalAlloc就是分别用于在局部堆或全局堆中分配内存。
由于每个进程的局部堆很小,所以在局部堆中分配内存会受到空间的限制。但这个堆是每个进程私有的,相对而言分配数据较安全,数据访问出错不至于影响到整个系统。 而在全局堆中分配的内存是为各个进程共享的,每个进程只要拥有这个内存块的句柄都可以访问这块内存,但是每个全局内存空间需要额外的内存开销,造成分配浪费。而且一旦发生严重错误,可能会影响到整个系统的稳定。
不过在Win32中,每个进程都只拥有一个缺省的私有堆,它只能被当前进程访问。应用程序也不可能直接访问系统内存。所以在Win32中全局堆和局部堆都指向进程的省缺堆。用LocalAlloc/GlobalAlloc分配内存没有任何区别。甚至LocalAlloc分配的内存可以被GlobalFree释放掉。所以在Win32下编程,无需注意Local和Global的区别。
-----------------------------------------------------------------------------------------------
看来,现在用 GlobalAlloc() 与 malloc() 分配内存都没有问题。