青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

小默

[zz]MFC - 深入探討MFC消息循環(huán)和消息泵(一)

首先,應該清楚MFC的消息循環(huán)(::GetMessage,::PeekMessage),消息泵(CWinThread::PumpMessage)和MFC的消息在窗口之間的路由是兩件不同的事情。在MFC的應用程序中(應用程序類基于CWinThread繼承),必須要有一個消息循環(huán),他的作用是從應用程序的消息隊列中讀取消息,并把它派送出去(::DispatchMessage)。而消息路由是指消息派送出去之后,系統(tǒng)(USER32.DLL)把消息投遞到哪個窗口,以及以后消息在窗口之間的傳遞是怎樣的。

消息分為隊列消息(進入線程的消息隊列)和非隊列消息(不進入線程的消息隊列)。對于隊列消息,最常見的是鼠標和鍵盤觸發(fā)的消息,例如WM_MOUSERMOVE,WM_CHAR等消息;還有例如:WM_PAINT、WM_TIMER和WM_QUIT。當鼠標、鍵盤事件被觸發(fā)后,相應的鼠標或鍵盤驅動程序就會把這些事件轉換成相應的消息,然后輸送到系統(tǒng)消息隊列,由Windows系統(tǒng)負責把消息加入到相應線程的消息隊列中,于是就有了消息循環(huán)(從消息隊列中讀取并派送消息)。還有一種是非隊列消息,他繞過系統(tǒng)隊列和消息隊列,直接將消息發(fā)送到窗口過程。例如,當用戶激活一個窗口系統(tǒng)發(fā)送WM_ACTIVATE, WM_SETFOCUS, and WM_SETCURSOR。創(chuàng)建窗口時發(fā)送WM_CREATE消息。在后面你將看到,MS這么設計是很有道理的,以及他的整套實現(xiàn)機制。

  這里講述MFC的消息循環(huán),消息泵。先看看程序啟動時,怎么進入消息循環(huán)的:

_tWinMain ->AfxWinMain ->AfxWinInit ->CWinThread::InitApplication ->CWinThread::InitInstance ->CWinThread::Run


  非對話框程序的消息循環(huán)的事情都從這CWinThread的一Run開始...

  第一部分:非對話框程序的消息循環(huán)機制。

//thrdcore.cpp
// main running routine until thread exits
int CWinThread::Run()
{
ASSERT_VALID(this);

// for tracking the idle time state
BOOL bIdle = TRUE;
LONG lIdleCount = 0;

// acquire and dispatch messages until a WM_QUIT message is received.
for (;;)
{
  // phase1: check to see if we can do idle work
  while (bIdle &&
   !::PeekMessage(&m_msgCur, NULL, NULL, NULL, PM_NOREMOVE))
  {
   // call OnIdle while in bIdle state
   if (!OnIdle(lIdleCount++))
    bIdle = FALSE; // assume "no idle" state
  }

  // phase2: pump messages while available
  do
  {
   // pump message, but quit on WM_QUIT
   if (!PumpMessage())
    return ExitInstance();

   // reset "no idle" state after pumping "normal" message
   if (IsIdleMessage(&m_msgCur))
   {
    bIdle = TRUE;
    lIdleCount = 0;
   }

  } while (::PeekMessage(&m_msgCur, NULL, NULL, NULL, PM_NOREMOVE));
}    //無限循環(huán),退出條件是收到WM_QUIT消息。

ASSERT(FALSE);  // not reachable
}

這是一個無限循環(huán),他的退出條件是收到WM_QUIT消息:



if (!PumpMessage())
    return ExitInstance();

   在PumpMessage中,如果收到WM_QUIT消息,那么返回FALSE,所以ExitInstance()函數(shù)執(zhí)行,跳出循環(huán),返回程序的退出代碼。所以,一個程序要退出,只用在代碼中調用函數(shù)

  VOID PostQuitMessage( int nExitCode )。指定退出代碼nExitCode就可以退出程序。

  下面討論一下這個函數(shù)Run的流程,分兩步:

  1,第一個內循環(huán)phase1。bIdle代表程序是否空閑。他的意思就是,如果程序是空閑并且消息隊列中沒有要處理的消息,那么調用虛函數(shù)OnIdle進行空閑處理。在這個處理中將更新UI界面(比如工具欄按鈕的enable和disable狀態(tài)),刪除臨時對象(比如用FromHandle得到的對象指針。由于這個原因,在函數(shù)之間傳遞由FromHandle得到的對象指針是不安全的,因為他沒有持久性)。OnIdle是可以重載的,你可以重載他并返回TRUE使消息循環(huán)繼續(xù)處于空閑狀態(tài)。

  NOTE:MS用臨時對象是出于效率上的考慮,使內存有效利用,并能夠在空閑時自動撤銷資源。關于由句柄轉換成對象,可以有若干種方法。一般是先申明一個對象obj,然后使用obj.Attatch來和一個句柄綁定。這樣產生的對象是永久的,你必須用obj.Detach來釋放對象。

  2,第二個內循環(huán)phase2。在這個循環(huán)內先啟動消息泵(PumpMessage),如果不是WM_QUIT消息,消息泵將消息發(fā)送出去(::DispatchMessage)。消息的目的地是消息結構中的hwnd字段所對應的窗口。



//thrdcore.cpp
BOOL CWinThread::PumpMessage()
{
ASSERT_VALID(this);

//如果是WM_QUIT就退出函數(shù)(return FALSE),這將導致程序結束.
if (!::GetMessage(&m_msgCur, NULL, NULL, NULL)) {
#ifdef _DEBUG
  if (afxTraceFlags & traceAppMsg)
   TRACE0("CWinThread::PumpMessage - Received WM_QUIT.\n");
  m_nDisablePumpCount++; // application must die
   // Note: prevents calling message loop things in 'ExitInstance'
   // will never be decremented
#endif
  return FALSE;
}

#ifdef _DEBUG
if (m_nDisablePumpCount != 0)
{
  TRACE0("Error: CWinThread::PumpMessage called when not permitted.\n");
  ASSERT(FALSE);
}
#endif

#ifdef _DEBUG
if (afxTraceFlags & traceAppMsg)
  _AfxTraceMsg(_T("PumpMessage"), &m_msgCur);
#endif

// process this message

if (m_msgCur.message != WM_KICKIDLE && !PreTranslateMessage(&m_msgCur))





{
  ::TranslateMessage(&m_msgCur); //鍵轉換
  ::DispatchMessage(&m_msgCur); //派送消息
}
return TRUE;
}

?
  在這一步有一個特別重要的函數(shù)大家一定認識:PreTranslateMessage。這個函數(shù)在::DispatchMessage發(fā)送消息到窗口之前,進行對消息的預處理。PreTranslateMessage函數(shù)是CWinThread的成員函數(shù),大家重載的時候都是在View類或者主窗口類中,那么,它是怎么進入別的類的呢?代碼如下:



//thrdcore.cpp
BOOL CWinThread::PreTranslateMessage(MSG* pMsg)
{
ASSERT_VALID(this);

// 如果是線程消息,那么將會調用線程消息的處理函數(shù)
if (pMsg->hwnd == NULL && DispatchThreadMessageEx(pMsg))
  return TRUE;

// walk from target to main window
CWnd* pMainWnd = AfxGetMainWnd();
if (CWnd::WalkPreTranslateTree(pMainWnd->GetSafeHwnd(), pMsg))
  return TRUE;

// in case of modeless dialogs, last chance route through main
//   window's accelerator table
if (pMainWnd != NULL)
{
   CWnd* pWnd = CWnd::FromHandle(pMsg->hwnd);
   if (pWnd->GetTopLevelParent() != pMainWnd)
   return pMainWnd->PreTranslateMessage(pMsg);
}

return FALSE;   // no special processing
}


  由上面這個函數(shù)可以看出:

  第一,如果(pMsg->hwnd == NULL),說明這是一個線程消息。調用CWinThread::DispatchThreadMessageEx到消息映射表找到消息入口,然后調用消息處理函數(shù)。

  NOTE: 一般用PostThreadMessage函數(shù)發(fā)送線程之間的消息,他和窗口消息不同,需要指定線程id,消息激被系統(tǒng)放入到目標線程的消息隊列中;用ON_THREAD_MESSAGE( message, memberFxn )宏可以映射線程消息和他的處理函數(shù)。這個宏必須在應用程序類(從CWinThread繼承)中,因為只有應用程序類才處理線程消息。如果你在別的類(比如視圖類)中用這個宏,線程消息的消息處理函數(shù)將得不到線程消息。

  第二,消息的目標窗口的PreTranslateMessage函數(shù)首先得到消息處理權,如果函數(shù)返回FALSE,那么他的父窗口將得到消息的處理權,直到主窗口;如果函數(shù)返回TRUE(表示消息已經被處理了),那么就不需要調用父類的PreTranslateMessage函數(shù)。這樣,保證了消息的目標窗口以及他的父窗口都可以有機會調用PreTranslateMessage--在消息發(fā)送到窗口之前進行預處理(如果自己處理完然后返回FALSE的話 -_-b),如果你想要消息不傳遞給父類進行處理的話,返回TRUE就行了。

  第三,如果消息的目標窗口和主窗口沒有父子關系,那么再調用主窗口的PreTranslateMessage函數(shù)。為什么這樣?由第二步知道,一個窗口的父窗口不是主窗口的話,盡管它的PreTranslateMessage返回FALSE,主窗口也沒有機會調用PreTranslateMessage函數(shù)。我們知道,加速鍵的轉換一般在框架窗口的PreTranslateMessage函數(shù)中。

  我找遍了MFC中關于加速鍵轉換的處理,只有CFrameWnd,CMDIFrameWnd,CMDIChildWnd等窗口類有。所以,第三步的意思是,如果消息的目標窗口(他的父窗口不是主窗口,比如一個這樣的非模式對話框)使消息的預處理繼續(xù)漫游的話(他的PreTranslateMessage返回FALSE),那么給一次機會給主窗口調用PreTranslateMessage(萬一他是某個加速鍵消息呢?),這樣能夠保證在有非模式對話框的情況下還能保證主窗口的加速鍵好使。
http://dev.csdn.net/article/51/51474.shtm

posted on 2009-11-13 10:55 小默 閱讀(778) 評論(0)  編輯 收藏 引用 所屬分類: Windows

導航

統(tǒng)計

留言簿(13)

隨筆分類(287)

隨筆檔案(289)

漏洞

搜索

積分與排名

最新評論

閱讀排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            免费不卡中文字幕视频| 亚洲最新中文字幕| 亚洲第一黄网| 国产精品麻豆成人av电影艾秋| 欧美亚洲三区| 99在线热播精品免费| 另类天堂视频在线观看| 亚洲手机在线| 一区二区三区日韩精品| 欧美在线视频免费| 亚洲欧洲综合另类| 久久国产精品免费一区| 亚洲欧美三级伦理| 在线观看日韩| 国产精品a级| 国产午夜精品久久| 国产精品福利在线观看| 欧美中文在线观看国产| 久久成人免费视频| 久色婷婷小香蕉久久| 亚洲天堂av电影| 一区二区日韩欧美| 在线视频免费在线观看一区二区| 久久久综合香蕉尹人综合网| 亚洲欧美自拍偷拍| 性欧美超级视频| 久久久精品一区二区三区| 久久久久久免费| 免费亚洲网站| 欧美不卡高清| 亚洲婷婷综合色高清在线 | 久久精品一区二区三区不卡牛牛| 久久天天躁狠狠躁夜夜爽蜜月 | 久久综合影视| 亚洲欧美在线高清| 欧美高潮视频| 国内精品模特av私拍在线观看| 亚洲美女91| 久久国产毛片| 国产精品久久久亚洲一区 | 久久精品视频免费观看| 亚洲精品在线视频| 久久伊人免费视频| 欧美高潮视频| 亚洲韩日在线| 久久午夜羞羞影院免费观看| 久久免费国产| 国语自产偷拍精品视频偷 | 午夜在线精品偷拍| 国产麻豆精品视频| 香蕉亚洲视频| 欧美一区二区视频免费观看| 狠狠色综合网| 欧美在线国产| 久久综合亚洲社区| 中文一区在线| 久久爱www.| 亚洲伦理久久| 欧美亚洲综合在线| 亚洲国产一区在线观看| 国产亚洲精品久久久久动| 六月天综合网| 国产欧美日韩精品a在线观看| 久久久噜噜噜久噜久久| 欧美国产综合一区二区| 亚洲淫性视频| 欧美激情精品久久久久久黑人| 亚洲欧美中文日韩v在线观看| 看欧美日韩国产| 国产精品久久久久久妇女6080| 欧美视频一区二| 牛牛影视久久网| 国产精品人成在线观看免费 | 原创国产精品91| 亚洲韩国精品一区| 国产亚洲综合精品| 在线视频精品一区| 欧美中文在线免费| 欧美成人综合网站| 亚洲综合丁香| 亚洲精品社区| 亚洲成色777777在线观看影院| 亚洲国产精品悠悠久久琪琪| 欧美日韩第一区| 欧美大片专区| 亚洲国产精品ⅴa在线观看| 99www免费人成精品| 亚洲综合精品| 精品电影一区| 欧美日韩另类一区| 午夜天堂精品久久久久 | 国产午夜亚洲精品羞羞网站| 亚洲一区二区av电影| 欧美一区三区三区高中清蜜桃 | 一区二区三区视频观看| 久久国产精品第一页| 亚洲日韩视频| 国产资源精品在线观看| 欧美精品成人一区二区在线观看| 国产精品99久久久久久久久| 91久久久在线| 亚洲国产精品一区二区尤物区| 午夜久久资源| 一区二区三区欧美在线| 亚洲天堂网在线观看| 欧美一级视频| 91久久国产综合久久| 国产精品99久久久久久人| 国产精品久久久久av| 欧美成人日本| 亚洲福利视频在线| 久久精品视频导航| 久久综合伊人77777麻豆| 黄网动漫久久久| 久久久久久久波多野高潮日日| 亚洲欧美伊人| 中国av一区| 亚洲激情av在线| 一区二区亚洲| 黄网动漫久久久| 一区二区视频免费在线观看| 国产日本欧美一区二区三区| 国产精品视频久久久| 国产精品美女在线| 国产亚洲毛片| 亚洲精品精选| 亚洲免费在线视频一区 二区| 亚洲天堂av综合网| 午夜影院日韩| 久久一二三四| 亚洲精品1区2区| 亚洲视屏一区| 久久久久国产精品厨房| 欧美成人午夜激情视频| 欧美日韩一区二区三区免费| 国产精品红桃| 在线日韩成人| 性做久久久久久| 亚洲国产免费| 午夜免费电影一区在线观看| 美女久久网站| 国产日韩欧美成人| 日韩性生活视频| 久热精品在线| 欧美在线视频观看免费网站| 午夜精品久久久久久久久久久| 欧美不卡一区| 欧美在线观看视频| 国产精品久久久久9999高清| 亚洲精品视频在线观看网站| 久久精品国产欧美亚洲人人爽| 亚洲人成人99网站| 欧美激情国产日韩| 亚洲黄色片网站| 欧美国产欧美综合 | 亚洲精品系列| 欧美激情综合| 欧美激情视频一区二区三区在线播放 | 久久色在线播放| 国产乱码精品一区二区三区不卡 | 免费视频久久| 国产欧美日韩伦理| 久久激情视频久久| 久久久伊人欧美| 亚洲福利小视频| 亚洲人成亚洲人成在线观看图片| 亚洲第一主播视频| 国产主播一区| 麻豆91精品91久久久的内涵| 欧美日韩网址| 亚洲人线精品午夜| 亚洲精品国产视频| 欧美激情久久久久| 亚洲激情一区二区| 亚洲第一中文字幕在线观看| 亚洲欧美春色| 久久婷婷蜜乳一本欲蜜臀| 激情成人亚洲| 欧美91大片| av72成人在线| 久久久国产精品一区二区中文| 日韩网站在线看片你懂的| 亚洲精品视频在线观看免费| 欧美日韩高清免费| 亚洲欧美另类久久久精品2019| 久久久欧美精品| 午夜亚洲性色视频| 亚洲国产日本| 欧美一区二区免费| 国产在线精品一区二区中文 | 欧美日韩国产黄| 午夜精品久久久久久| 亚洲电影激情视频网站| 亚洲欧美一级二级三级| 国产一区再线| 国产精品户外野外| 另类国产ts人妖高潮视频| 制服诱惑一区二区| 卡一卡二国产精品| 午夜精品一区二区三区电影天堂|