電子發(fā)燒友App

硬聲App

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫(xiě)文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示
創(chuàng)作
電子發(fā)燒友網(wǎng)>電子資料下載>嵌入式開(kāi)發(fā)>MFC中Windows消息處理機(jī)制解析

MFC中Windows消息處理機(jī)制解析

2017-11-07 | rar | 0.2 MB | 次下載 | 1積分

資料介紹

Windows程序和DOS程序的主要不同點(diǎn)之一是:Windows程序是以事件為驅(qū)動(dòng)、消息機(jī)制為基礎(chǔ)
  本人對(duì)Windows系統(tǒng)、MFC談不上有深入的了解,但對(duì)MFC本身包裝API的機(jī)制很有興趣,特別是讀了候老師的《深入淺出MFC》后,感覺(jué)到Visual C++的Application FrameWork十分精制。在以前,我對(duì)SDI結(jié)構(gòu)處理消息有一定的認(rèn)識(shí),但對(duì)于模式對(duì)話框的消息機(jī)制不了解,讀了《深入》一書(shū)也沒(méi)能得到解決,近日,通過(guò)在網(wǎng)友的幫助和查閱MSDN,自認(rèn)為已經(jīng)了解。一時(shí)興起,寫(xiě)下這些文字,沒(méi)有其它目的,只是希望讓后來(lái)者少走彎路,也希望和我一樣 喜歡“鉆牛角尖”的人共同討論、學(xué)習(xí)。如果你是牛人,那么你現(xiàn)在要慎重考慮有沒(méi)有充足的時(shí)間讀這些幼稚文字。
  正文:
  Windows程序和DOS程序的主要不同點(diǎn)之一是:Windows程序是以事件為驅(qū)動(dòng)、消息機(jī)制為基礎(chǔ)。如何理解?
  舉了例子,當(dāng)你CLICK Windows “開(kāi)始”BUTTON時(shí),為什么就會(huì)彈出一個(gè)菜單呢?
  當(dāng)你單擊鼠標(biāo)左鍵時(shí),操作系統(tǒng)中與MOUSE相關(guān)的驅(qū)動(dòng)程序在第一時(shí)間內(nèi)得到這個(gè)信號(hào)[LBUTTONDOWN],然后它通知操作系統(tǒng)―――“嗨,鼠標(biāo)左鍵被單擊了!”,操作系統(tǒng)得到這一信號(hào)后,馬上要判斷――用戶(hù)單擊鼠標(biāo)左鍵,這是針對(duì)哪個(gè)窗口呢?如何判斷?這很簡(jiǎn)單!當(dāng)前狀態(tài)中,具有焦點(diǎn)的窗口[或控件]就是了[這里當(dāng)然是“開(kāi)始”BUTTON了]。然后操作系統(tǒng)馬上向這個(gè)窗口發(fā)送一條消息到這個(gè)窗口所在進(jìn)程的消息隊(duì)列,消息內(nèi)容應(yīng)是消息本身的代號(hào)、附加參數(shù)、窗口句柄…等等了。那么,只有操作系統(tǒng)才有資格發(fā)送消息至某一窗口的消息隊(duì)列嗎?不然,其它程序也有資格。你可以在你的程序中調(diào)用:SendMessage、PostMessage。這樣,被單擊的窗口得到了一條由操作系統(tǒng)發(fā)送的包含CLICK的消息,操作系統(tǒng)已經(jīng)暫時(shí)不再管窗口的任何事,因?yàn)樗€要忙于處理其它事務(wù)。你的程序得到一條消息后如何做呢?Windows對(duì)于你在“開(kāi)始”BUTTON上的單擊事件做出如下反映:彈出一菜單。可是,得到消息到做出反映這一過(guò)程是如何實(shí)現(xiàn)的呢?這就是本文討論的主要內(nèi)容[當(dāng)然只是針對(duì)MFC了]。
  我首先簡(jiǎn)要談一下SDI,然后會(huì)花更多文字描述模式對(duì)話框。
  對(duì)于SDI窗口,你的應(yīng)用程序類(lèi)的InitInstance()大約如下:
  BOOL CEx06aApp::InitInstance()
  {
  ……………
  CSingleDocTemplate* pDocTemplate;
  pDocTemplate = new CSingleDocTemplate(
  IDR_MAINFRAME,
  RUNTIME_CLASS(CEx06aDoc),
  RUNTIME_CLASS(CMainFrame), // main SDI frame window
  RUNTIME_CLASS(CEx06aView));
  AddDocTemplate(pDocTemplate);
  CCommandLineInfo cmdInfo;
  ParseCommandLine(cmdInfo);
  if (!ProcessShellCommand(cmdInfo))
  return FALSE;
  m_pMainWnd-》ShowWindow(SW_SHOW);
  m_pMainWnd-》UpdateWindow();
  return TRUE;
  }
  完成一些如動(dòng)態(tài)生成相關(guān)文檔、視,顯示主框架窗口、處理參數(shù)行信息等工作。這些都是顯示在你工程中的“明碼”。我們現(xiàn)在把斷點(diǎn)設(shè)置到return TRUE;一句,跟入MFC源碼中,看看到底MFC內(nèi)部做了什么。
  程序進(jìn)入SRCWinMain.cpp,下一個(gè)大動(dòng)作應(yīng)是:
  nReturnCode = pThread-》Run();
  注意了,重點(diǎn)來(lái)了。F11進(jìn)入
  int CWinApp::Run()
  {
  if (m_pMainWnd == NULL && AfxOleGetUserCtrl())
  {
  // Not launched /Embedding or /Automation, but has no main window!
  TRACE0(“Warning: m_pMainWnd is NULL in CWinApp::Run - quitting application. ”);
  AfxPostQuitMessage(0);
  }
  return CWinThread::Run();
  }
  再次F11進(jìn)入:
  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));
  }
  ASSERT(FALSE); // not reachable
  }
  BOOL CWinThread::IsIdleMessage(MSG* pMsg)
  {
  // Return FALSE if the message just dispatched should _not_
  // cause OnIdle to be run. Messages which do not usually
  // affect the state of the user interface and happen very
  // often are checked for.
  // redundant WM_MOUSEMOVE and WM_NCMOUSEMOVE
  if (pMsg-》message == WM_MOUSEMOVE || pMsg-》message == WM_NCMOUSEMOVE)
  {
  // mouse move at same position as last mouse move?
  if (m_ptCursorLast == pMsg-》pt && pMsg-》message == m_nMsgLast)
  return FALSE;
  m_ptCursorLast = pMsg-》pt; // remember for next time
  m_nMsgLast = pMsg-》message;
  return TRUE;
  }
  // WM_PAINT and WM_SYSTIMER (caret blink)
  return pMsg-》message != WM_PAINT && pMsg-》message != 0x0118;
  }
  這是SDI處理消息的中心機(jī)構(gòu),但請(qǐng)注意,它覺(jué)對(duì)不是核心!
  分析一下,在無(wú)限循環(huán)FOR內(nèi)部又出現(xiàn)一個(gè)WHILE循環(huán)
  while (bIdle &&
  ?。海篜eekMessage(&m_msgCur, NULL, NULL, NULL, PM_NOREMOVE))
  {
  // call OnIdle while in bIdle state
  if (!OnIdle(lIdleCount++))
  bIdle = FALSE; // assume “no idle” state
  }
  這段代碼是當(dāng)你程序進(jìn)程的消息隊(duì)列中沒(méi)有消息時(shí),會(huì)調(diào)用OnIdle做一些后備工作,臨時(shí)對(duì)象在這里被刪除。當(dāng)然它是虛函數(shù)。其中的PeekMessage,是查看消息隊(duì)列,如果有消息返回TRUE,如果沒(méi)有消息返回FALSE,這里指定PM_NOREMOVE,是指查看過(guò)后不移走消息隊(duì)列中剛剛被查看
  到的消息,也就是說(shuō)這里的PeekMessage只起到一個(gè)檢測(cè)作用,顯然返回FALSE時(shí)[即沒(méi)有消息],才會(huì)進(jìn)入循環(huán)內(nèi)部,執(zhí)行OnIdle,當(dāng)然了,你的OnIdle返回FLASE,會(huì)讓程序不再執(zhí)行OnIdle。你可能要問(wèn):
  當(dāng)bidle=0或消息隊(duì)例中有消息時(shí),程序又執(zhí)行到哪了呢?
  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));
  看啊,又進(jìn)入一個(gè)循環(huán)!
  其中有個(gè)重要的函數(shù),PumpMessage,內(nèi)容如下:
  BOOL CWinThread::PumpMessage()
  {
  ASSERT_VALID(this);
  if (?。海篏etMessage(&m_msgCur, NULL, NULL, NULL))
  {
  #ifdef _DEBUG
  if (afxTraceFlags & traceAppMsg)
  TRACE0(“CWinThread::PumpMessage - Received WM_QUIT. ”);
  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. ”);
  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);
 ?。海篋ispatchMessage(&m_msgCur);
  }
  return TRUE;
  }
  如你所想,這才是MFC消息處理的核心基地[也是我個(gè)人認(rèn)為的]。
  GetMessage不同于PeekMessae,它是不得到消息不罷體,PeekMessage如果發(fā)現(xiàn)消息隊(duì)列中沒(méi)有消息會(huì)返回0,而GetMessage如果發(fā)現(xiàn)沒(méi)有消息,等,直到有了消息,而且,GetMessage不同于PeekMessage,它會(huì)將消息移走[當(dāng)然,PeekMessage也可以做到這點(diǎn)]。我想當(dāng)你讀了這個(gè)函數(shù)后,
  你應(yīng)明白PreTranslateMessage函數(shù)的用法了吧[我比較喜歡在程序中充分利用這個(gè)函數(shù)]。
 ?。海篢ranslateMessage(&m_msgCur);
 ?。海篋ispatchMessage(&m_msgCur);
  將消息發(fā)送到窗口的處理函數(shù)[它是由窗口類(lèi)指定的],之后的動(dòng)作一直到你的程序做出反映的過(guò)程,你可以在《深入》一書(shū)中得到完美的解釋。我們還是通過(guò)reurn
  TRUE;回到CWinThread::Run()中的Do{}while;循環(huán)。然后還是對(duì)IDLE的處理,即便剛才你的ONIDLE返回了FALSE,在這里你看到,你的程序還是有機(jī)會(huì)執(zhí)行它的。然后又是利用PeekMessage檢測(cè)消息隊(duì)列:
  如果有消息[這個(gè)消息不被移動(dòng)的原因是因?yàn)樗獮镻umpMessage內(nèi)的GetMessage所利用。]再次進(jìn)入PumpMessage[叫它“消息泵”吧]。
  如果沒(méi)有消息,退出DO循環(huán),但它還在FOR內(nèi)部,所以又執(zhí)行第一個(gè)While循環(huán)。
下載該資料的人也在下載 下載該資料的人還在閱讀
更多 >

評(píng)論

查看更多

下載排行

本周

  1. 1TC358743XBG評(píng)估板參考手冊(cè)
  2. 1.36 MB  |  330次下載  |  免費(fèi)
  3. 2開(kāi)關(guān)電源基礎(chǔ)知識(shí)
  4. 5.73 MB  |  6次下載  |  免費(fèi)
  5. 3100W短波放大電路圖
  6. 0.05 MB  |  4次下載  |  3 積分
  7. 4嵌入式linux-聊天程序設(shè)計(jì)
  8. 0.60 MB  |  3次下載  |  免費(fèi)
  9. 5基于FPGA的光纖通信系統(tǒng)的設(shè)計(jì)與實(shí)現(xiàn)
  10. 0.61 MB  |  2次下載  |  免費(fèi)
  11. 6基于FPGA的C8051F單片機(jī)開(kāi)發(fā)板設(shè)計(jì)
  12. 0.70 MB  |  2次下載  |  免費(fèi)
  13. 751單片機(jī)窗簾控制器仿真程序
  14. 1.93 MB  |  2次下載  |  免費(fèi)
  15. 8基于51單片機(jī)的RGB調(diào)色燈程序仿真
  16. 0.86 MB  |  2次下載  |  免費(fèi)

本月

  1. 1OrCAD10.5下載OrCAD10.5中文版軟件
  2. 0.00 MB  |  234315次下載  |  免費(fèi)
  3. 2555集成電路應(yīng)用800例(新編版)
  4. 0.00 MB  |  33564次下載  |  免費(fèi)
  5. 3接口電路圖大全
  6. 未知  |  30323次下載  |  免費(fèi)
  7. 4開(kāi)關(guān)電源設(shè)計(jì)實(shí)例指南
  8. 未知  |  21548次下載  |  免費(fèi)
  9. 5電氣工程師手冊(cè)免費(fèi)下載(新編第二版pdf電子書(shū))
  10. 0.00 MB  |  15349次下載  |  免費(fèi)
  11. 6數(shù)字電路基礎(chǔ)pdf(下載)
  12. 未知  |  13750次下載  |  免費(fèi)
  13. 7電子制作實(shí)例集錦 下載
  14. 未知  |  8113次下載  |  免費(fèi)
  15. 8《LED驅(qū)動(dòng)電路設(shè)計(jì)》 溫德?tīng)栔?/a>
  16. 0.00 MB  |  6653次下載  |  免費(fèi)

總榜

  1. 1matlab軟件下載入口
  2. 未知  |  935054次下載  |  免費(fèi)
  3. 2protel99se軟件下載(可英文版轉(zhuǎn)中文版)
  4. 78.1 MB  |  537796次下載  |  免費(fèi)
  5. 3MATLAB 7.1 下載 (含軟件介紹)
  6. 未知  |  420026次下載  |  免費(fèi)
  7. 4OrCAD10.5下載OrCAD10.5中文版軟件
  8. 0.00 MB  |  234315次下載  |  免費(fèi)
  9. 5Altium DXP2002下載入口
  10. 未知  |  233046次下載  |  免費(fèi)
  11. 6電路仿真軟件multisim 10.0免費(fèi)下載
  12. 340992  |  191185次下載  |  免費(fèi)
  13. 7十天學(xué)會(huì)AVR單片機(jī)與C語(yǔ)言視頻教程 下載
  14. 158M  |  183278次下載  |  免費(fèi)
  15. 8proe5.0野火版下載(中文版免費(fèi)下載)
  16. 未知  |  138040次下載  |  免費(fèi)