大概区别: PostMessage只负责将消息放到消息队列中,不确定何时及是否处理 SendMessage要等到受到消息处理的返回码(DWord类型)后才继续 PostMessage执行后马上返回 SendMessage必须等到消息被处理后才会返回。
详细说明:
函数原型:LRESULT SendMessage(HWND hWnd,UINT Msg,WPARAM wParam,LPARAM IParam);
参数:
hWnd:其窗口程序将接收消息的窗口的句柄。如果此参数为HWND_BROADCAST,则消息将被发送到系统中所有顶层窗口,包括无效或不可见的非自 身拥有的窗口、被覆盖的窗口和弹出式窗口,但消息不被发送到子窗口。
Msg:指定被发送的消息。
wParam:指定附加的消息指定信息。
IParam:指定附加的消息指定信息。
返回值:返回值指定消息处理的结果,依赖于所发送的消息。
备注:需要用HWND_BROADCAST通信的应用程序应当使用函数RegisterWindowMessage来为应用程序间的通信取得一个唯一的消 息。
如果指定的窗口是由调用线程创建的,则窗口程序立即作为子程序调用。如果指定的窗口是由不同线程创建的,则系统切换到该线程并调用恰当的窗口程序。线程间 的消息只有在线程执行消息检索代码时才被处理。发送线程被阻塞直到接收线程处理完消息为止。
Win32 API消息函数:PostMessage
函数功能:该函数将一个消息放入(寄送)到与指定窗口创建的线程相联系消息队列里,不等待线程处理消息就返回。消息队列里的消息通过调用 GetMessage和PeekMessage取得。
函数原型:B00L PostMessage(HWND hWnd,UINT Msg,WPARAM wParam,LPARAM lParam);
参数
hWnd:其窗口程序接收消息的窗口的句柄。可取有特定含义的两个值:
HWND.BROADCAST:消息被寄送到系统的所有顶层窗口,包括无效或不可见的非自身拥有的窗口、被覆盖的窗口和弹出式窗口。消息不被寄送到子窗 口。
NULL:此函数的操作和调用参数dwThread设置为当前线程的标识符PostThreadMessage函数一样。
Msg:指定被寄送的消息。
wParam:指定附加的消息特定的信息。
IParam:指定附加的消息特定的信息。
返回值:如果函数调用成功,返回非零值:如果函数调用失败,返回值是零。若想获得更多的错误信息,请调用GetLastError函数。
备注:需要以 HWND_BROADCAST方式通信的应用程序应当用函数 RegisterwindwosMessage来获得应用程序间通信的独特的消息。
如果发送一个低于WM_USER范围的消息给异步消息函数 (PostMessage.SendNotifyMessage,SendMesssgeCallback),消息参数不能包含指针。否则,操作将会失 败。函数将再接收线程处理消息之前返回,发送者将在内存被使用之前释放。
在控制别的应用程序的时候,经常需要等待直到某个功能结束,例如: 打开一个窗口-->等待直到窗口结束 这 个时候就可以用到SendMessage 如果在打开这个窗口后仍然需要对该窗口的界面进行设置,比如Edit的value等等,比如: 打 开一个窗口-->控制窗口的control的属性 这个时候就需要PostMessage 使用一个钩子程序截获消息后,使用 SendMessage把消息发送到主处理程序进行处理,但是在主处理程序还没有完成任务的时候,被设置钩子的程序进入了停止的状态,不可以处理 WM_PAINT, WM_MOVE, ....... 等的基本信息, 必须要等SendMessage发送出的消息完成后,才能继续运行,整个界面一片空白,把钩子消息设置成PostMessage的 发送消息形式后,问题解决! 查 MSDN对这两个API的定义, PostMessage只是把消息放入队列,不管其他程序是否处理都返 回,然后继续执行; 而SendMessage必须等待其他程序处理消息后才返回,继续执行。 PostMessage的返回值表示 PostMessage函数执行是否正确; 而SendMessage的返回值表示其他程序处理消息后的返回值。 使用这两个发送消息函数 的最重要的是要看你的程序是否要对消息的滞后性关注否,PostMessage会造成消息的滞后性,而SendMessage则不会,但如果 SendMessage消息处理失败,则会造成程序停止!
===============================================
1. SendMessage函数要等到消息被处理后才返回
PostMessage 消息发出后马上返回
2. PostMessage发向目标窗口的消息一定会进入消息队列
SendMessage向同一线程的窗口发消息,不会进入消息队列,
SendMessage向其他线程的窗口发消息,则会进入消息队列,
3. PostMessage的返回值表示PostMessage函数执行是否正确,
SendMessage的返回值表示其他程序处理消息后的返回值
1、SendMessage()和PostMessage().这两个函数几乎是一样的,它们的区别是:SendMessage()直接把一个消息发给窗口过程,等消息被处理后才返回。Postmessage()只是把消息发送到消息队列,完成后即返回。
2.1、当调用SendMessage时,接收消息的线程的QS_SENDMESSAGE标志被设置,系统调用相应的窗口过程来处理这个消息。 GetMessage和PeekMessage函数在内部进行这种处理。如果在“发送消息队列”中没有消息了,QS_SENDMESSAGE标志就被关 闭。“发送消息队列”中可能有几个Send过来的消息。例如,几个线程同时向一个窗口发送消息。
2.2、当调用PostMessage时,函数GetMessage或PeekMessage填充传递给它们的Msg结构,函数返回。再调用 DispatchMessage,让相应的窗口过程来处理消息。
2.3、当调用SendMessage时,发送线程和接收线程是同一个线程的时候,SendMessag很简单,只是调用指定窗口的窗口过程。但当 发送线程和接收线程不是同一个线程时,麻烦就大了。因为发送线程和接收线程运行在不同的地址空间中,因此不能访问接受线程中相应窗口过程的代码和数据。其 实这时发送线程要挂起,当接收线程处理Send过来的消息时,发送线程被设置为空闲(idle)状态。在发送的消息处理完成后,窗口过程的返回值被登记到 发送线程的应答消息队列中。发送线程被唤醒,取出包含在应答消息队列中的返回值。这个返回值就是调用SendMessage的返回值,这时,发送线程继续 运行。
2.4、WM_PAINT和WM_TIMER这两个消息的优先级非常低,最低的是WM_TIMER。
2.5、WM_COPYDATA只能Send,不能Post。
PostMessage(hWnd,WM_CLOSE,NULL,NULL) //PostMessage,只是简单的把WM_CLOSE放在消息队列里,然后函数立刻返回, //继续执行下面的代码,那我们要执行的WM_CLOSE什么时候执行呢?这个系统说了算 //系统会在合适的时候从消息队列里取出WM_CLOSE,在代码里体现为 //GetMessage返回,参数msg.message设为WM_CLOSE,之后如代码所示,会执行到 //DispatchMessage,DispatchMessage内部调用SendMessage,SendMessage内部调用 //窗口函数WndProc,最后才会跳到 “case WM_CLOSE”
........ ........ SendMessage(hWnd,WM_PASTE,NULL,NULL) //SendMessage内部调用了WndProc,之后就会跳转到"case WM_PASTE" ,再之后WndProc返回, //SendMessage返回。程序才会继续往下执行。从理解的角度来看,你完全可以把 //SendMessage(hWnd,WM_PASTE,NULL,NULL) 等同于调用函数 //WndProc(hWnd,WM_PASTE,NULL,NULL) ,这样就很好理解了。
二、总结来说: ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 1. PostMessage(WM_CLOSE)简单把消息放入队列并返回-> GetMessage->DispatchMessage->SendMessage->WndProc->处理消息WM_CLOSE
2. SendMessage(WM_CLOSE) 就相当于调用WndProc(WM_CLOSE)
三、PostMessage、SendMessage和 GetMessage 内部结构简化: ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
PostMessage内部结构简化: BOOL PostMessage(HWND hWnd,UINT Msg,WPARAM wParam,LPARAM lParam) { .................... 把消息放到窗口消息队列中 .................... return TRUE; }
SendMessage内部结构简化: LRESULT SendMessage(HWND hWnd,UINT Msg,WPARAM wParam,LPARAM lParam) { .................... .................... LRESULT ret=WndProc(hWnd,Msg,wParam,lParam); .................... .................... return ret; }
GetMessage内部结构简化: BOOL GetMessage(LPMSG lpMsg,HWND hWnd,wMsgFilterMin,wMsgFilterMax) { ....................
GetMessage是一个很神奇的函数,它不是按我们常规那样理解的。 常规函数里的代码是一行一行往下执行并最后返回,但GetMessage里有堵塞结构, 系统如果发现消息队列里没有消息,就让代码停下来,整个程序也就没有“运动了” (后台程序往往处于这种情况) 不停地查看消息队列里是否有消息,如果没有就停在这里,不往下执行。 如果发现有消息了,比如系统发现鼠标点击了窗口,会把WM_LBUTTONDOWN 放到消息队列里,这时就可以从队列中取出WM_LBUTTONDOWN消息,继续往下执行
....................
lpMsg->message=WM_LBUTTONDOWN;
....................
return TRUE; //GetMessage基本都是返回TRUE,否则while循环就推出,程序也就结束了 }