当前位置: 代码迷 >> 综合 >> GetMessage PeekMessage SendMessage PostMessage
  详细解决方案

GetMessage PeekMessage SendMessage PostMessage

热度:9   发布时间:2023-12-22 06:05:49.0

GetMessage () and PeekMessage ()
GetMessage 其唯一用途就是从事件对列中获得消息,并进行处理。当程序在等待通过 GetMessage 传递的消息时,主事件循坏基本上是锁定的。
那,如果我们要处理实时的事件循坏该怎么办呢?Windows 为我们提供了另一个函数 PeekMessage,他们的原型几乎是一样的,唯一的不同是 PeekMessage 比 GetMessage 多了一个参数:UINT wRemoveMsg // removal flags。
对于wRemoveMsg ,有效的标志有
PM_NOREMOVE,PeekMessage 处理之后,消息没有从序列中去除
PM_REMOVE,PeekMessage 处理之后,消息已经从序列中去除
怎样来实现实时事件循坏呢?
通过利用PeekMessage 来判断消息序列中是否有消息,如果有,就处理它;否则继续处理其他的游戏逻辑并重复进行。
他们的函数原型如下:
BOOL GetMessage (
LPMSG lpMsg, // pointer to structure for message
HWND hWnd, // handle to window
UINT wMsgFilterMin, // first message
UINT wMsgFilterMax ) // last message
BOOL PeekMessage (
LPMSG lpMsg, // pointer to structure for message
HWND hWnd, // handle to window
UINT wMsgFilterMin, // first message
UINT wMsgFilterMax, // last message
UINT wRemoveMsg ); // removal flags
SendMessage () and PostMessage ()
SendMessage 向窗口传递一个要求立即处理的消息。接收窗口处理完该消息后,该函数便紧接着 WinProc 返回。是同步的。
PostMessage 将消息发往窗口的消息序列,而后直接返回。如果不在意在消息被处理以前的时间延迟,或者该消息的优先级较低,就可以使用该函数。是异步的。
这两个函数的原型几乎一样,唯一的不同是返回值不同。SendMessage 返回 LRESULT,PostMessage 返回的是 BOOL。他们的函数原型如下:
LRESULT SendMessage ( HWND hWnd, UINT Msg, WPARAM wParam, r
LPARAM lParam);  非队列消息
BOOL PostMessage ( HWND hWnd, UINT Msg, WPARAM wParam, 

LPARAM lParam); 队列消息

 

typedef struct tagMSG { 
    HWND hwnd; //接收消息的窗口句柄  UINT message; //消息标识(ID)
    WPARAM wParam; LPARAM lParam; 
    DWORD time;// 消息产生的时间  POINT pt;// 消息产生时鼠标的位置
} MSG; 
键盘、鼠标消息以外,队列消息还有WM_PAINT、WM_TIMER和WM_QUIT。 
while (GetMessage(&msg, (HWND) NULL, 0, 0)) {//从消息队列得到消息 
       if (hwndDlgModeless == (HWND) NULL || !IsDialogMessage(hwndDlgModeless, &msg) && !TranslateAccelerator(hwndMain, haccel, &msg)) {  //不是对话框消息和快捷键消息
     TranslateMessage(&msg); 
      DispatchMessage(&msg); //发送消息 
} } 
 

当得到消息WM_QUIT,或者::GetMessage出错时,退出消息循环。
系统消息ID的范围是从0到WM_USER-1,或0X80000到0XBFFFF;应用程序消息从WM_USER(0X0400)到0X7FFF,或 0XC000到0XFFFF;WM_USER到0X7FFF范围的消息由应用程序自己使用;0XC000到0XFFFF范围的消息用来和其他应用程序通信,为了ID的唯一性,使用::RegisterWindowMessage来得到该范围的消息ID。