WM_CLOSE、WM_DESTROY、WM_QUIT及各種消息投遞函數(shù)詳解
本文對(duì)WM_CLOSE、WM_DESTROY、WM_QUIT及各種消息投遞函數(shù)的功能及區(qū)別做出了分析比對(duì),有助于讀者更好的對(duì)消息投遞函數(shù)加以理解。詳情如下:
一、WM_CLOSE、WM_DESTROY、WM_QUIT區(qū)別
WM_CLOSE:關(guān)閉應(yīng)用程序窗口
WM_DESTROY:關(guān)閉應(yīng)用程序
WM_QUIT:關(guān)閉消息循環(huán)
只有關(guān)閉了消息循環(huán),應(yīng)用程序的進(jìn)程才真正退出(在任務(wù)管理器里消失)。
win32應(yīng)用程序的完整退出過(guò)程:點(diǎn)擊窗口右上角的關(guān)閉按鈕,發(fā)送WM_CLOSE消息。此消息處理中調(diào)用DestroyWindow函數(shù),發(fā)送WM_DESTROY消息。此消息處理中調(diào)用PostQuitMessage(0)函數(shù),發(fā)送WM_QUIT消息到消息隊(duì)列中。GetMessage捕獲到WM_QUIT,返回0,退出循環(huán)(應(yīng)用程序真正退出)。
注意:按照上述正常流程,WM_QUIT是不會(huì)到達(dá)窗口過(guò)程的。(因?yàn)樵贕etMessage截獲了WM_QUIT消息之后,程序已經(jīng)徹底退出了!)
MFC應(yīng)用程序的完整退出過(guò)程:點(diǎn)擊窗口右上角的關(guān)閉按鈕,或選擇【File/Close】,發(fā)出 WM_CLOSE消息。CMyFrameWnd 并沒(méi)有設(shè)置WM_CLOSE 處理常式,于是交給預(yù)設(shè)之處理常式。預(yù)設(shè)函數(shù)對(duì)于WM_CLOSE 的處理方式是呼叫 ::DestroyWindow, 并因而發(fā)出WM_DESTROY。預(yù)設(shè)之WM_DESTROY 處理方式是呼叫::PostQuitMessage,因此發(fā)出WM_QUIT。CWinApp::Run 收到WM_QUIT 后會(huì)結(jié)束其內(nèi)部之訊息回路, 然后呼叫ExitInstance,這是CWinApp 的一個(gè)虛擬函數(shù)。如果自己應(yīng)用程序類CMyWinApp 改寫了ExitInstance , 那么CWinApp::Run 所呼叫的就是CMyWinApp::ExitInstance,否則就是 CWinApp::ExitInstance。最后回到 AfxWinMain,執(zhí)行 AfxWinTerm,結(jié)束程序。
附加:當(dāng)調(diào)用DestroyWindow函數(shù)后,操作系統(tǒng)就會(huì)進(jìn)行一系列的刪除動(dòng)作,先發(fā)送WM_DESTROY消息,接著發(fā)送WM_NCDESTROY消息。如果這個(gè)窗口還有子窗口或者是其它窗口的所有者,就需要給所有子窗口發(fā)送刪除消息。
WM_QUIT是唯一可以使GetMessage(&msg,NULL,0,0)返回假值的消息.
相關(guān)代碼分析:
//主函數(shù)中進(jìn)入消息循環(huán)的代碼片斷 while(GetMessage(&msg,NULL,0,0)) { TranslateMessage(&msg); //將消息進(jìn)行處理一下 DispatchMessage(&msg); //再將消息變量msg傳給windows,讓windows來(lái)調(diào)用消息處理函數(shù) }
如果把GetMessage(&msg,NULL,0,0)改為GetMessage(&msg,hWnd,0,0),則發(fā)現(xiàn)關(guān)閉應(yīng)用程序后,任務(wù)管理器中仍有該程序的進(jìn)程,且占用大量的內(nèi)存,why?
msdn中的原因解釋是:對(duì)于GetMessage(&msg,hWnd,0,0),當(dāng)?shù)诙€(gè)參數(shù)無(wú)效時(shí),此函數(shù)返回值為-1。對(duì)于上述循環(huán)來(lái)說(shuō),此while條件為真,因此進(jìn)入死循環(huán),進(jìn)程無(wú)法退出。
二、各種消息投遞函數(shù)
1、SendMessage:發(fā)送消息給指定的窗口過(guò)程;直到窗口過(guò)程處理了消息才返回。
2、PostMessage:將消息放入消息隊(duì)列(與指定窗口創(chuàng)建的線程相關(guān))中;無(wú)需等待消息處理,立即返回。
不能發(fā)送WM_QUIT消息,此消息只能由PostQuitMessage函數(shù)發(fā)送。
3、PostThreadMessage:發(fā)送消息給指定線程的消息隊(duì)列;無(wú)需等待線程處理消息,立即返回。
此函數(shù)發(fā)送的消息和窗口是無(wú)關(guān)的。我們只需指定線程ID就OK了,但要保證線程已創(chuàng)建,否則會(huì)失敗。
4、GetMessage:從調(diào)用線程的消息隊(duì)列中取消息。
當(dāng)?shù)诙€(gè)參數(shù)為NULL時(shí),它檢索以下兩種消息:
a、屬于調(diào)用線程的任何窗口的消息;
b、由PostThreadMessag投遞給該調(diào)用線程的消息。
5、PeekMessage:功能同GetMessage。區(qū)別是:
GetMessage:直到一個(gè)匹配了過(guò)濾條件的消息,被放到消息隊(duì)列中才返回。
PeekMessage:不管消息是否放入隊(duì)列,立即返回。
6、DispatchMessage:派遣消息給相應(yīng)的窗口過(guò)程。
7、TranslateMessage:轉(zhuǎn)換虛擬鍵信息到字符消息。
相關(guān)文章
Qt6.0+vs2019環(huán)境配置的實(shí)現(xiàn)教程
這篇文章主要介紹了Qt6.0+vs2019環(huán)境配置的實(shí)現(xiàn)教程,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-03-03C語(yǔ)言實(shí)現(xiàn)掃雷游戲詳細(xì)代碼實(shí)例
這篇文章主要介紹了C語(yǔ)言實(shí)現(xiàn)掃雷游戲詳細(xì)代碼實(shí)例,有感興趣的同學(xué)可以借鑒參考下2021-02-02C語(yǔ)言單鏈表實(shí)現(xiàn)圖書管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言單鏈表實(shí)現(xiàn)圖書管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03QT 中文亂碼解決匯總(QString與string、char*互轉(zhuǎn)亂碼)
在QT中使用中文時(shí),經(jīng)常會(huì)碰到論碼問(wèn)題,本文主要介紹了QT 中文亂碼解決匯總(QString與string、char*互轉(zhuǎn)亂碼),需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-07-07C++ STL入門教程(7) multimap、multiset的使用
這篇文章主要介紹了C++ STL入門教程第七篇,multimap一對(duì)多索引,multiset多元集合的使用方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-08-08C語(yǔ)言的數(shù)據(jù)結(jié)構(gòu)之樹、森連、二叉樹之間的轉(zhuǎn)換圖解
這篇文章主要介紹了C語(yǔ)言的數(shù)據(jù)結(jié)構(gòu)之樹、森連、二叉樹之間的轉(zhuǎn)換詳解,數(shù)據(jù)是信息的載體,是描述客觀事物屬性的數(shù)、字符以及所有能輸入到計(jì)算機(jī)中并被程序識(shí)別和處理的符號(hào)的集合,需要的朋友可以參考下2023-07-07C語(yǔ)言動(dòng)態(tài)內(nèi)存管理的原理及實(shí)現(xiàn)方法
C語(yǔ)言動(dòng)態(tài)內(nèi)存管理的原理是通過(guò) malloc() 函數(shù)申請(qǐng)一塊連續(xù)的內(nèi)存空間,并返回其地址,通過(guò) free() 函數(shù)釋放該內(nèi)存空間。實(shí)現(xiàn)方法是通過(guò)在程序運(yùn)行時(shí)動(dòng)態(tài)地管理內(nèi)存,即在需要內(nèi)存時(shí)申請(qǐng),不需要時(shí)釋放,避免了靜態(tài)內(nèi)存分配的浪費(fèi)和不足2023-04-04