PostMessage
功能说明
PostMessage将消息投递到进程的消息队列,不会等待进程处理消息完毕就返回,PostMessage所投递的消息会被进程所调用的GetMessage或者PeekMessage函数所检索。PostMessage对消息的投递行为属于异步发送,即发送消息而不必等待消息被处理之后再返回。PostMessage在投递消息时,系统根据hWnd参数来判断所属进程而投递,投递到创建指定窗口的进程的消息队列中去。
跟SendMessage区别:SendMessage将一个消息发送到接收窗口后不会立即返回,而是等待消息的处理结果,所以SendMessage不但关心消息是否发送成功,而且也要关心消息是否被窗口处理完毕,也就是说SendMessage是同步的。
函数原型
B00L PostMessage(
HWND hWnd, // 消息的投递方向
UINT Msg, // 所发消息之消息ID
WPARAM wParam,// 所发消息之字参数
LPARAM lParam // 所发消息之值参数
);
参数
1>、hWnd,【in】
消息的投递方向,可取有特定含义的两个值:
Value
描述
HWND_BROADCAST
消息的投递方向为:系统的所有顶层窗口,包括无效的、不可见的无属配的窗口以及被覆盖的窗口和弹出式窗口,而不包含它们的子窗口。
NULL
函数行为跟调用PostThreadMessage(其参数dwThread为当前线程ID)一样。
2>、Msg,【in】
指定的要投递的消息。
3>、wParam,【in】
发送消息的字参数,跟消息类型、性质相关。
4>、lParam,【in】
发送消息的值参数,跟消息类型、性质相关。
返回值
如果函数调用成功(消息投递成功),返回非零值;如果函数调用失败,返回值是零
备注
需要以 HWND_BROADCAST方式投递消息的应用程序,应当用函数 RegisterwindwosMessage来获得应用程序间通信的特征消息。
Win32下:系统仅仅为系统级消息进行封送处理(那些定义在0到WM_USER 范围内的消息),要发送自定义消息(那些数值定义在WM_USER之上的消息)到另一个进程,用户必须进行自定义封送处理(就是编组处理之意)。
如果发送一个系统级消息(在0到WM_USER范围内)给异步消息函数(PostMessage.SendNotifyMessage,SendMesssgeCallback),消息参数不能包含指针,否则,操作将会失败。因为函数在接收进程对消息处理完毕之前就返回了,并且消息发送者也会在内存被使用之前就释放掉了。
不要使用PostMessage函数投递WM_QUIT消息,要使用PostQuitMessage函数。
Windows 2000/XP: 每个消息队列有10,000个投递消息数量限制。然而这个限制可以根据实际需求被设置为充分的大,如果你的应用程序对投递消息的需求量超过这个限制,应当重新设计,以避免消耗太多的系统资源。如果要调整这个限制,需要修改注册表项如下:
HKEY_LOCAL_MACHINE
SOFTWARE
Microsoft
Windows NT
CurrentVersion
Windows
USERPostMessageLimit
可接受的最低值为4000。
适用
Windows NT: 3.1及以上版本;Windows:95及以上版本;Windows CE:1.0及以上版本;头文件:winuser.h;输入库:user32.lib;Unicode:在Windows NT环境下以Unicode和ANSI方式实现。
应用举例
客户端的应用程序可以使用DDE建立一个跟服务端应用程序中某条目(或者叫做项目)的链接,当这样一个链接建立起来之后,服务端周期性地更新被客户端链接的条目(项目)的内容,通常是数值变化(引起客户端同样的变化)。这样就在两个应用之间建立起了永久性的数据流向,一直保持到服务显示地断开。客户端应用程序一般通过投递消息WM_DDE_ADVISE来初始化这条数据链,如下代码所示:
if (!(hOptions = GlobalAlloc(GMEM_MOVEABLE, sizeof(DDEADVISE)))) return;
if (!(lpOptions = (DDEADVISE FAR*) GlobalLock(hOptions))) { GlobalFree(hOptions); return; }
lpOptions->cfFormat = CF_TEXT;
lpOptions->fAckReq = TRUE;
lpOptions->fDeferUpd = FALSE;
GlobalUnlock(hOptions);
if ((atomItem = GlobalAddAtom(szItemName)) != 0) {
if ( !( PostMessage(hwndServerDDE, WM_DDE_ADVISE, (WPARAM)hwndClientDDE,
PackDDElParam(WM_DDE_ADVISE, (UINT) hOptions, atomItem) ) ) ) {
GlobalDeleteAtom(atomItem);
GlobalFree(hOptions);
FreeDDElParam(WM_DDE_ADVISE, lParam);
}
}
if (atomItem == 0) { // Handle errors }
见MSDN2005之ms-help://MS.MSDNQTR.v80.en/MS.MSDN.v80/MS.WIN32COM.v10.en/winui/winui/windowsuserinterface/dataexchange/dynamicdataexchange/usingdynamicdataexchange.htm
.版本 2
.DLL命令 寻找子窗口_, 整数型, “user32”, “FindWindowExA”, , , 在窗口列表中寻找与指定条件相符的第一个子窗口 找到的窗口的句柄。如未找到相符窗口,则返回零。会设置GetLastError
.参数 窗口句柄1, 整数型, , 在其中查找子的父窗口。如设为零,表示使用桌面窗口(通常说的顶级窗口都被认为是桌面的子窗口,所以也会对它们进行查找)
.参数 窗口句柄2, 整数型, , 从这个窗口后开始查找。这样便可利用对FindWindowEx的多次调用找到符合条件的所有子窗口。如设为零,表示从第一个子窗口开始搜索
.参数 欲搜索的类名1, 文本型, , 欲搜索的类名。“”表示忽略
.参数 欲搜索的类名2, 文本型, , 欲搜索的类名。“”表示忽略;