PeekMessage、GetMessage的区别

在Windows编程中经常使用这两个函数来处理消息,它们之间的区别就是GetMessage是阻塞的,PeekMessage是非阻塞的。

GetMessage原型如下:BOOL GetMessage(LPMSG lpMsg,HWND hWnd,UINT
wMsgFilterMin,UINT wMsgFilterMax);

PeekMessage原型如下:BOOL PeekMessage(LPMSG lpMsg,HWND hWnd,UINT
wMsgFilterMin,UINT wMsgFilterMax,UINT wRemoveMsg);

前面4个参数都是一样的,第一个是消息结构指针;第二个是窗口句柄,标识要接收的消息的窗口,如果为NULL,则表示接收属于该应用程序的所有窗口的消息;第三个、第四个分别指定了接收的消息的范围,如果都为0,则接收所有消息,WM_KEYFIRST
和 WM_KEYLAST 常量用于接受所有的键盘消息。 WM_MOUSEFIRST 和 WM_MOUSELAST
常量用于接受所有的鼠标消息。PeekMessage函数的第五个参数指示了PeekMeesage接收过消息之后是否将该消息删除,PM_REMOVE表示删除,PM_NONREMOVE表示不删除。

该函数用于查看应用程序的消息队列,如果其中有消息就将其放入lpMsg所指的结构中,不过,与GetMessage不同的是,
PeekMessage函数不会等到有消息放入队列时才返回。同样,如果hWnd为NULL,则PeekMessage获取属于调用该函数应用程序的任一窗口的消息,如果hWnd=-1,那么函数只返回把hWnd参数为NULL的PostAppMessage函数送去的消息。如果
wMsgFilterMin和wMsgFilterMax都是0,则PeekMessage就返回所有可得到的消息。函数获取之后将删除消息队列中的除
WM_PAINT消息之外的其他消息,至于WM_PAINT则只有在其处理之后才被删除。

Windows编程中常用的消息处理函数如下:

1 while(GetMessage(&msg, NULL, 0, 0))
2 {
3 TranslateMessage(&msg);
4 DispatchMessage(&msg);
5 }

首先,GetMessage从进程的主线程的消息队列中获取一个消息并将它复制到MSG结构,如果队列中没有消息,则GetMessage函数将等待一个消息的到来以后才返回。如果你将一个窗口句柄作为第二个参数传入GetMessage,那么只有指定窗口的的消息可以从队列中获得。GetMessage也可以从消息队列中过滤消息只接受消息队列中落在范围内的消息。这时候就要利用GetMessage/PeekMessage指定一个消息过滤器。这个过滤器是一个消息标识符的范围或者是一个窗体句柄,或者两者同时指定。当应用程序要查找一个后入消息队列的消息是很有用。WM_KEYFIRST
和 WM_KEYLAST 常量用于接受所有的键盘消息。 WM_MOUSEFIRST 和 WM_MOUSELAST 常量用于接受所有的鼠标消息。

TranslateMessage将把两个按键消息WM_KEYDOWN和WM_KEYUP
转换成一个WM_CHAR,不过需要注意的是,消息WM_KEYDOWN,WM_KEYUP仍然将传递给窗口的回调函数。

处理完之后,DispatchMessage函数将把此消息发送给该消息指定的窗口中已设定的回调函数。如果消息是WM_QUIT,则
GetMessage返回0,从而退出循环体。应用程序可以使用PostQuitMessage来结束自己的消息循环。通常在主窗口的
WM_DESTROY消息中调用。

使用PeekMessage函数的消息循环如下:


 1 while(true)
2 {
3 if(PeekMessage(&msg, m_hWnd, WM_KEYFIRST, WM_KEYLAST, PM_REMOVE))
4 {
5 if (msg.message == WM_QUIT)
6 {
7 break;
8 }
9 if(msg.message == WM_KEYDWON && msg.wParam == VK_ESCAPE)
10 {
11 ...
12 }
13 }
14 else
15 {
16 // no message
17 }
18 }

这里我们接受所有的键盘消息,所以就用WM_KEYFIRST 和
WM_KEYLAST作为参数。最后一个参数可以是PM_NOREMOVE 或者 PM_REMOVE,表示消息信息是否应该从消息队列中删除。

所以这段小代码就是判断是否按下了Esc键,如果是就进行处理。

注意:

while (PeekMessage (&msg, NULL, 0, 0, PM_REMOVE)) ;

这行叙述从消息伫列中删除WM_PAINT之外的所有消息。如果伫列中有一个WM_PAINT消息,程式就会永远地陷在while循环中。

时间: 2024-08-03 11:51:48

PeekMessage、GetMessage的区别的相关文章

实例说明 PeekMessage与GetMessage的区别

PeekMessage与GetMessage的对比相同点:PeekMessage函数与GetMessage函数都用于查看应用程序消息队列,有消息时将队列中 的消息派发出去. 不同点:无论应用程序消息队列是否有消息,PeekMessage函数都立即返回,程序得以继续执行 后面的语句(无消息则执行其它指令,有消息时一般要将消息派发出去,再执行其它 指令).GetMessage函数只有在消息对立中有消息时返回,队列中无消息就会一直等,直至下 一个消息出现时才返回.在等的这段时间,应用程序不能执行任何指

如何区别PeekMessage&GetMessage SendMessage&PostMessage

转自http://blog.csdn.net/young0325/article/details/6430664 Peekmessage和Getmessage都是向系统的消息队列中取得消息,不过性质不同. 若第一次向消息队列中取不到消息,则程序的主线程会被OS(操作系统)挂起;等到OS重新调度到该线程时,而且消息队列仍然是空的时,两者的性质不同: 若使用Getmessage(),则程序的主线程会仍被OS挂起. 若是用Peekmessage(),则程序会取得OS控制权,运行一段时间.此函数多用于处

PeekMessage&GetMessage

原文:http://www.cnblogs.com/faceang/archive/2010/05/25/1743757.html PeekMessage与GetMessage的对比相同点:PeekMessage函数与GetMessage函数都用于查看应用程序消息队列,有消息时将队列中 的消息派发出去. 不同点:无论应用程序消息队列是否有消息,PeekMessage函数都立即返回,程序得以继续执行 后面的语句(无消息则执行其它指令,有消息时一般要将消息派发出去,再执行其它 指令).GetMess

VC++中PostMessage、SendMessage和PeekMessage之间的区别

1, PostMessage只把消息放入队列,不管其他程序是否处理都返回,然后继续执行,这是个异步消息投放函数.而SendMessage必须等待其他程序处理消息完了之后才返回,继续执行,这是个同步消息投放函数.而且,PostMessage的返回值表示PostMessage函数执行是否正确:而SendMessage的返回值表示其他程序处理消息后的返回值.这点大家应该都明白. 2, 如果在同一个线程内,PostMessage发送消息时,消息要先放入线程的消息队列,然后通过消息循环Dispatch到目

Java中printStackTrace()、toString()、getMessage()的区别

一.三者之间的关系图: 二.演示 1.printStackTrace()演示: public class Test {     public int div(int a, int b)     {         try         {             return a/b;         } catch (Exception e)         {            e.printStackTrace();         }         return 0;     }

PreTranslateMessage和TranslateMessage区别

PreTranslateMessage是消息在送给TranslateMessage函数之前被调用的,绝大多数本窗口的消息都要通过这里,比较常用,当需要在MFC之前处理某些消息时,常常要在这里添加代码. MFC 消息控制流最具特色的地方是CWnd类的虚拟函数PreTranslateMessage(),通过重载这个函数,可以改变MFC的消息控制流程,甚至可 以作一个全新的控制流出来.只有穿过消息队列的消息才受PreTranslateMessage()影响,采用SendMessage()或其他类似的方

PeekMessage究竟做了什么?

1.UI线程 2.工作线程 把Delphi里TThread的WaitFor函数转化成C++代码,就会是下面这个样子. BOOL TThread::WaitFor(HANDLE hThread) { MSG msg; HANDLE handle[1]; handle[0] = hThread; DWORD dwWaitResult = 0; do { // This prevents a potential deadlock if the background thread // does a S

Windows消息队列

一 Windows中有一个系统消息队列,对于每一个正在执行的Windows应用程序,系统为其建立一个"消息队列",即应用程序队列,用来存放该程序可能 创建的各种窗口的消息.应用程序中含有一段称作"消息循环"的代码,用来从消息队列中检索这些消息并把它们分发到相应的窗口函数中.  二 Windows为当前执行的每个Windows程序维护一个「消息队列」.在发生输入事件之后,Windows将事件转换为一个「消息」并将消息放入程序的消息队列中.程序通过执行一块称之为「消息循

Win32/MFC的基本概念

一.MFC的基本概念 单文档.多文档和对话框框架的区别 MFC中的类继承图的基本框架 CView类与CDocument的关系 Onpaint()和Ondraw()的关系 hdc-cdc区别联系 RUNTIME_CLASS是什么 DECLARE_DYNAMIC 和 IMPLEMENT_DYNAMIC DECLARE_DYNCREATE 和 IMPLEMENT_DYNCREATE DECLARE_SERIAL 和 IMPLEMENT_SERIAL RUNTIME_CLASS宏的定义是这样的: #de