何謂消息、消息處理函數、消息映射? 消息簡單的說就是指通過輸入設備向程序發出指令要執行某個操作。具體的某個操作是你的一系列代碼。稱為消息處理函數。
在SDK中消息其實非常容易理解,當窗口建立后便會有一個函數(窗口處理函數)開始執行一個消息循環,我們還可以清楚的看到消息處理的脈絡。一個switch case語句就可以搞定,消息循環直到遇到WM_QUIT消息才會結束,其余的消息均被攔截后調用相應的處理函數。
但在封裝了API的MFC中,消息似乎變的有些復雜了,我們看不到熟悉的switch case語句了,取而代之的是一個叫消息映射的東西。為什么MFC要引入消息映射機制,你可以想象一下,在現在的程序開發活動中,你的一個程序是否擁有多個窗體,主窗口就算只有一個,那菜單、工具條、控件這些都是子窗口,那我們需要寫多少個switch case,並且還要為每個消息分配一個消息處理函數,這樣做是多么的復雜呀。因此MFC采用了一種新的機制。利用一個數組,將窗口消息和相對應的消息處理函數進行映射,你可以理解成這是一個表。這種機制就是消息映射。我們看不到熟悉的switch case語句了,取而代之的是一個叫消息映射的東西。為什么MFC要引入單因此MFC采用了一種新的機制。利用一個數組,將窗口消息和相對應的消息處理函數進行映射,你可以理解成這是一個表。這張表在窗口基類CWnd定義,派生類的消息映射表如果你沒有動作它是空的,也就是說如果你不手工的增加消息處理函數,則當派生窗口接受一個消息時會執行父類的消息處理函數。這樣做顯然是高效的。
MFC提供的消息結構 同時MFC定義了下面的兩個主要結構: AFX_MSGMAP_ENTRY struct AFX_MSGMAP_ENTRY{ UINT nMessage; // Windows消息的ID號 UINT nCode; // 控制消息的通知 UINT nID; // Windows控制消息的ID UINT nLastID; //表示是一個指定范圍的消息被映射的范圍 UINT nSig; //表示消息的動作標識 AFX_PMSG pfn; // 指向消息處理函數的指針 }; AFX_MSGMAP struct AFX_MSGMAP{ #ifdef _AFXDLL const AFX_MSGMAP* (PASCAL* pfnGetBaseMap)(); #else const AFX_MSGMAP* pBaseMap; #endif const AFX_MSGMAP_ENTRY* lpEntries; }; ///AFX_MSGMAP可以得到基類的消息映射入口地址和得到本身的消息映射入口地址。
MFC下一個消息的處理過程是一般是這樣的。 1、_AfxCbtFilterHook截獲消息(這是一個鈎子函數) 2、_AfxCbtFilterHook把窗口過程設定為AfxWndProc。 3、函數AfxWndProc接收Windows操作系統發送的消息。 4、函數AfxWndProc調用函數AfxCallWndProc進行消息處理。 5、函數AfxCallWndProc調用CWnd類的方法WindowProc進行消息處理。
如何添加自己的消息? 我們已經了解了WINDOW的消息機制,如何加入我們自己的消息呢?好我們來看 一個標准的消息處理程序是這個樣子的 在 CWnd 類中預定義了標准 Windows 消息 (WM_XXXX WM是WINDOW MESSAGE的縮寫) 的默認處理程序。類庫基於消息名命名這些處理程序。例如,WM_PAINT 消息的處理程序在 CWnd 中被聲明為: afx_msg void OnPaint(); afx_msg 關鍵字通過使這些處理程序區別於其他 CWnd 成員函數來表明 C++ virtual 關鍵字的作用。但是請注意,這些函數實際上並不是虛擬的,而是通過消息映射實現的。我們在本文的一開始便說明了為什么要這樣做。 所有能夠進行消息處理的類都是基於CCmdTarget類的,也就是說CCmdTarget類是所有可以進行消息處理類的父類。CCmdTarget類是MFC處理命令消息的基礎和核心。
若要重寫基類中定義的處理程序,只需在派生類中定義一個具有相同原型的函數,並創建此處理程序的消息映射項。我們通過ClassWizard可以建立大多數窗口消息或自定義的消息,通過ClassWizard可以自動建立消息映射,和消息處理函數的框架,我們只需要把我們要做的事情填空,添加你要做的事情到處理函數。這個非常簡單,就不細說了。但是也許我們需要添加一些ClassWizard不支持的窗口消息或自定義消息,那么就需要我們親自動手建立消息映射和消息處理的框架,通常步驟如下: 第一步:定義消息。Microsoft推薦用戶自定義消息至少是WM_USER+100,因為很多新控件也要使用WM_USER消息。 #define WM_MYMESSAGE (WM_USER + 100)
第二步:實現消息處理函數。該函數使用WPRAM和LPARAM參數並返回LPESULT。 LPESULT CMainFrame::OnMyMessage(WPARAM wParam, LPARAM lParam) { // TODO: 處理用戶自定義消息,填空就是要填到這里。 return 0; } 第三步:在類頭文件的AFX_MSG塊中說明消息處理函數: // {{AFX_MSG(CMainFrame) afx_msg LRESULT OnMyMessage(WPARAM wParam, LPARAM lParam); //}}AFX_MSG DECLARE_MESSAGE_MAP() 第四步:在用戶類的消息塊中,使用ON_MESSAGE宏指令將消息映射到消息處理函數中。 ON_MESSAGE( WM_MYMESSAGE, OnMyMessage )
可以看出,用戶自定義的消息和我們通過ClassWizard添加的消息一樣,都是利用了ON_MESSAGE宏,建立的消息映射。
其實消息類別可以分成多種,上面說的只是其中之一。有三種主要的消息類別:(以下部分摘自MSDN) 1、Windows 消息 此類消息主要包括以前綴 WM_ 開頭的消息,WM_COMMAND 除外。Windows 消息由窗口和視圖處理。此類消息往往帶有用於確定如何處理消息的參數。 2、控件通知 此類消息包括從控件和其他子窗口發送到其父窗口的 WM_COMMAND 通知消息。例如,當用戶在編輯控件 (Edit Control) 中執行可能更改文本的操作后,該編輯控件 (Edit Control) 將向其父級發送包含 EN_CHANGE 控件通知代碼的 WM_COMMAND 消息。該消息的窗口處理程序以某種適當的方式響應此通知消息,例如在控件中檢索該文本。 框架像傳送其他 WM_ 消息一樣傳送控件通知消息。但是有一個例外的情況,即當用戶單擊按鈕時由按鈕發送的 BN_CLICKED 控件通知消息。該消息被作為命令消息特別處理,並像其他命令一樣傳送。 3、命令消息 此類消息包括用戶界面對象(菜單、工具欄按鈕和快捷鍵)發出的 WM_COMMAND 通知消息。框架處理命令的方式與處理其他消息不同,可以使用更多種類的對象處理命令。 Windows 消息和控件通知消息由窗口來處理(窗口是從 CWnd 類派生的類的對象)。包括 CFrameWnd、CMDIFrameWnd、CMDIChildWnd、CView、CDialog 以及從這些基類派生的您自己的類。這些對象封裝了 HWND——Windows 窗口的句柄。 命令消息可以由范圍更廣的對象(文檔、文檔模板以及應用程序對象本身)處理,而不僅僅由窗口和視圖處理。當某一命令直接影響到某個特定對象時,應當讓該對象處理此命令。例如,“文件”菜單中的“打開”命令在邏輯上與應用程序相關聯:該應用程序接收到此命令時會打開指定的文檔。因此“打開”命令的處理程序是應用程序類的成員函數。
命令消息我們比較常見的便是菜單項和工具條了,大家可以看到他的消息映射宏和窗口消息不太一樣,一般的形式是這樣的 ON_COMMAND(id,memberFxn) 第一個參數是命令ID,一個ID號對應一個消息處理,當然你可以讓多個ID共用一個處理函數。常見的應用例如:菜單項打開文檔的ID和工具條按鈕打開文檔的ID同時使用一個處理函數,或者直接將它們的ID設成相同的。
還有一種消息叫通知消息。例如樹型控件的等一些復雜的控件在單擊后需要傳遞更多的信息,例如光標的位置和當前項的一個結構,所以MFC為控件的每個通知消息也定義了一個宏,它長成了這個樣子: ON_CONTROL(EN_CHANGE,id,memberFxn)
還有很多種消息存在於MFC,宏定義有區別,大家可以觸類旁通。
窗口消息有上百個。你可以從MSDN上查到WM_開頭的,或者查看CWnd的成員函數,會給你列出很多,別忘了還有很多非窗口消息。雷神無法一一列出,也沒有必要。大家查一下就行了。不過對一些常用的、新的控件消息和特殊的通知消息我還是把他們列出幾個表,大家做個參考吧。
MFC中處理消息的順序
1. AfxWndProc()接收消息,尋找消息所屬的CWnd對象,然后調用AfxCallWndProc( )。
2. AfxCallWndProc()存儲消息(消息標識符和消息參數)供未來參考,然后調用WindowProc( )。
3. WindowProc()發送消息給OnWndMsg( ),如果消息未被處理,則發送給DefWindowproc( )。
4. OnWndMsg()首先按字節對消息進行排序,對於WM_COMMAND消息,調用OnCommand()消息響應函數;對於WM_NOTIFY消息調用OnNotify()消息響應函數。任何被遺漏的消息將是標准消息。OnWndMsg()函數搜索類的消息映像,以找到一個能處理任何窗口消息的處理函數。如果OnWndMsg()函數不能找到這樣的處理函數的話,則把消息返回到WindowProc()函數,由它將消息發送給DefWindowProc()函數。
5. OnCommand()查看這是不是一個控件通知(lParam參數不為NULL),如果它是,OnCommand()函數會試圖將消息映射到制造通知的控件;如果它不是一個控件通知,或者控件拒絕映射的消息,OnCommand()就會調用OnCmdMsg()函數。
6. OnNotify( )也試圖將消息映射到制造通知的控件;如果映射不成功,OnNotify( )就調用相同的OnCmdMsg( )函數。
7. 根據接收消息的類,OnCmdMsg()函數將在一個稱為命令傳遞(Command Routing)的過程中潛在的傳遞命令消息和控件通知。例如:如果擁有該窗口的類是一個框架類,則命令和控件通知消息也被傳遞到視圖和文檔類,並為該類尋找一個消息處理函數。
MFC中創建窗口的順序
1. PreCreateWindow()是一個重載函數,在窗口被創建前,可以在該重載函數中改變創建參數(可以設置窗口風格等等)。
2. PreSubclassWindow()也是一個重載函數,允許首先子分類一個窗口OnGetMinMaxInfo()為消息響應函數,響應的是WM_GETMINMAXINFO消息,允許設置窗口的最大或者最小尺寸。
3. OnNcCreate()也是一個消息響應函數,響應WM_NCCREATE消息,發送消息以告訴窗口的客戶區即將被創建。
4. OnNcCalcSize()也是消息響應函數,響應WM_NCCALCSIZE消息,作用是允許改變窗口客戶區大小。
5. OnCreate()也是消息響應函數,響應WM_CREATE消息,發送消息告訴一個窗口已經被創建。
6. OnSize()也是消息響應函數,響應WM_SIZE消息,發送該消息以告訴該窗口大小已經發生變化。
7. OnMove()也是消息響應函數,響應WM_MOVE消息,發送此消息說明窗口在移動。
8. OnChildNotify()為重載函數,作為部分消息映射被調用,告訴父窗口即將被告知一個窗口剛剛被創建。
MFC中打開模態對話框的順序
1. DoModal()是重載函數,重載DoModal()成員函數。
2. PreSubclassWindow()也是重載函數,允許首先子分類一個窗口。
3. OnCreate()是消息響應函數,響應WM_CREATE消息,發送此消息以告訴一個窗口已經被創建。
4. OnSize()也是消息響應函數,響應WM_SIZE消息,發送此消息以告訴窗口大小發生變化。
5. OnMove()也是消息響應函數,響應WM_MOVE消息,發送此消息以告訴窗口正在移動。
6. OnSetFont()也是消息響應函數,響應WM_SETFONT消息,發送此消息以允許改變對話框中控件的字體。
7. OnInitDialog()也是消息響應函數,響應WM_INITDIALOG消息,發送此消息以允許初始化對話框中的控件,或者是創建新控件。
8. OnShowWindow()也是消息響應函數,響應WM_SHOWWINDOW消息,該函數被ShowWindow()函數調用。
9. OnCtlColor()也是消息響應函數,響應WM_CTLCOLOR消息,被父窗口發送已改變對話框或對話框上面控件的顏色。
10. OnChildNotify()是重載函數,作為WM_CTLCOLOR消息的結果發送。
MFC中關閉模態對話框的順序
1. OnClose()是消息響應函數,響應WM_CLOSE消息,當"關閉"按鈕被單擊的時候,該函數被調用。
2. OnKillFocus()也是消息響應函數,響應WM_KILLFOCUS消息,當一個窗口即將失去鍵盤輸入焦點以前被發送。
3. OnDestroy()也是消息響應函數,響應WM_DESTROY消息,當一個窗口即將被銷毀時,被發送。
4. OnNcDestroy()也是消息響應函數,響應WM_NCDESTROY消息,當一個窗口被銷毀以后被發送。
5. PostNcDestroy()也是重載函數,作為處理OnNcDestroy()函數的最后動作被CWnd調用。
MFC中打開非模態對話框的順序
1. PreSubclassWindow()是重載函數,允許用戶首先子分類一個窗口。
2. OnCreate()是消息響應函數,響應WM_CREATE消息,發送此消息以告訴一個窗口已經被創建。
3. OnSize()也是消息響應函數,響應WM_SIZE消息,發送此消息以告訴窗口大小發生變化。
4. OnMove()也是消息響應函數,響應WM_MOVE消息,發送此消息以告訴窗口正在移動。
5. OnSetFont()也是消息響應函數,響應WM_SETFONT消息,發送此消息以允許改變對話框中控件的字體。
MFC中關閉非模態對話框的順序
1. OnClose()是消息響應函數,響應窗口的WM_CLOSE消息,當關閉按鈕被單擊的時候發送此消息。
2. OnDestroy()也是消息響應函數,響應窗口的WM_DESTROY消息,當一個窗口將被銷毀時發送此消息。
3. OnNcDestroy()也是消息響應函數,響應窗口的WM_NCDESTROY消息,當一個窗口被銷毀后發送此消息。
4. PostNcDestroy()是重載函數,作為處理OnNcDestroy()函數的最后動作,被CWnd調用。
轉自:https://blog.csdn.net/zz709196484/article/details/76033033