標籤:
原文網址:http://blog.csdn.net/hyhnoproblem/article/details/6182646
本例通過在單文檔程式的視圖中添加WM_LBUTTONCLICK訊息處理函數,來解釋一般視窗訊息的投遞流程。 基於VS 2005
[cpp] view plaincopy
- BEGIN_MESSAGE_MAP(CMyView, CView)
- ON_WM_LBUTTONDBLCLK()
- END_MESSAGE_MAP()
- // ON_WM_LBUTTONDBLCLK宏展開
- #define ON_WM_LBUTTONDBLCLK() /
- { WM_LBUTTONDBLCLK, 0, 0, 0, AfxSig_vwp, /
- (AFX_PMSG)(AFX_PMSGW) /
- (static_cast< void (AFX_MSG_CALL CWnd::*)(UINT, CPoint) > ( &ThisClass :: OnLButtonDblClk)) },
從上面的代碼可以看出WM_LBUTTONCLICK訊息的類型標籤是AfxSig_vwp, 在afxmsg_.h中有一個枚舉類型AfxSig,訊息標籤主要用於區分訊息處理函數的類型。該類型中定義了AfxSig_vwp的值:
[cpp] view plaincopy
- enum AfxSig
- {
- //...
- AfxSig_vWp = AfxSig_v_W_p
- //...
- }
在AfxWndProc中,將訊息中的控制代碼映射成視窗類別指標,這個指標指向CMyView。AfxWndProc調用AfxCallWndProc,AfxCallWndProc調用CWnd::WindowProc,CWnd::WindowProc調用CWnd::OnWndMsg,CWnd::OnWndMsg完成對訊息的處理。
[cpp] view plaincopy
- // wincore.cpp 1746
- BOOL CWnd::OnWndMsg(UINT message, WPARAM wParam, LPARAM lParam, LRESULT* pResult)
- {
- // ...
- // GetMessageMap是個虛函數,因為當前指標是指向CMyView,所以取到的是CMyView的訊息映射表
- const AFX_MSGMAP* pMessageMap; pMessageMap = GetMessageMap();
- UINT iHash; iHash = (LOWORD((DWORD_PTR)pMessageMap) ^ message) & (iHashMax-1);
- winMsgLock.Lock(CRIT_WINMSGCACHE);
- // 全域訊息散列緩衝,尋找緩衝
- AFX_MSG_CACHE* pMsgCache; pMsgCache = &_afxMsgCache[iHash];
- const AFX_MSGMAP_ENTRY* lpEntry;
- if (message == pMsgCache->nMsg && pMessageMap == pMsgCache->pMessageMap)
- {
- // cache hit
- lpEntry = pMsgCache->lpEntry;
- winMsgLock.Unlock();
- if (lpEntry == NULL)
- return FALSE;
- // cache hit, and it needs to be handled
- if (message < 0xC000)
- goto LDispatch; // 系統訊息?
- else
- goto LDispatchRegistered; // 登入訊息?
- }
- else
- {
- // not in cache, look for it
- pMsgCache->nMsg = message;
- pMsgCache->pMessageMap = pMessageMap;
- for (/* pMessageMap already init‘ed */; pMessageMap->pfnGetBaseMap != NULL;
- pMessageMap = (*pMessageMap->pfnGetBaseMap)())
- {
- // Note: catch not so common but fatal mistake!!
- // BEGIN_MESSAGE_MAP(CMyWnd, CMyWnd)
- ASSERT(pMessageMap != (*pMessageMap->pfnGetBaseMap)());
- if (message < 0xC000)
- {
- // constant window message
- if ((lpEntry = AfxFindMessageEntry(pMessageMap->lpEntries,
- message, 0, 0)) != NULL)
- {
- pMsgCache->lpEntry = lpEntry;
- winMsgLock.Unlock();
- goto LDispatch;
- }
- }
- else
- {
- // registered windows message
- lpEntry = pMessageMap->lpEntries;
- while ((lpEntry = AfxFindMessageEntry(lpEntry, 0xC000, 0, 0)) != NULL)
- {
- UINT* pnID = (UINT*)(lpEntry->nSig);
- ASSERT(*pnID >= 0xC000 || *pnID == 0);
- // must be successfully registered
- if (*pnID == message)
- {
- pMsgCache->lpEntry = lpEntry;
- winMsgLock.Unlock();
- goto LDispatchRegistered;
- }
- lpEntry++; // keep looking past this one
- }
- }
- }
- pMsgCache->lpEntry = NULL;
- winMsgLock.Unlock();
- return FALSE;
- }
- // ...
- LDispatch:
- mmf.pfn = lpEntry->pfn;
- switch (lpEntry->nSig)
- {
- //...
- case AfxSig_v_u_p: // 訊息標籤
- {
- CPoint point(lParam);
- (this->*mmf.pfn_v_u_p)(static_cast<UINT>(wParam), point);
- }
- break;
- //...
- }
- //...
- LDispatchRegistered: // for registered windows messages
- ASSERT(message >= 0xC000);
- ASSERT(sizeof(mmf) == sizeof(mmf.pfn));
- mmf.pfn = lpEntry->pfn;
- lResult = (this->*mmf.pfn_l_w_l)(wParam, lParam);
- }
- // 訊息處理函數類型枚舉
- union MessageMapFunctions
- {
- AFX_PMSG pfn; // generic member function pointer
- // ...
- LRESULT (AFX_MSG_CALL CWnd::*pfn_l_w_l)(WPARAM, LPARAM);
- };
【轉】Windows訊息投遞流程:一般視窗訊息投遞(WM_LBUTTONCLICK)