af 框架流程

2021-06-19 16:22:13 字數 3613 閱讀 2026

原文出自:

了解mfc程式的啟動過程,對於初學者來講,了學習mfc很有幫助;對於不常用vc的人來說,過一段時間就會忘記。還是來記下來,方便以後查閱。

2、winmain登場

extern "c" int winapi

_twinmain(hinstance hinstance, hinstance hprevinstance, lptstr lpcmdline, int ncmdshow)

_twinmain函式的「_t」是為了支援unicode而準備的乙個巨集。

_twinmain函式返回值是afxwinmain函式的返回值,afxwinmain函式定義於winmain.cpp第21行,稍加整理,去蕪存菁,就可以看到這個「程式進入點」主要做些什麼事:

afxwininit(hinstance, hprevinstance, lpcmdline, ncmdshow);

afxwinterm();

return nreturncode;

}用過sdk寫程式的朋友,現在可能會發出會心的微笑。

3、afxwininit——afx內部初始化操作

相當於呼叫:

相當於呼叫:

6、cframewnd::create產生主視窗(並先註冊視窗類)

其中create是cframewnd的成員函式,它將產生乙個視窗,用過sdk程式設計序的朋友都知道,要建立主視窗時要先註冊乙個視窗類,規定視窗的屬性等,但,這裡使用哪乙個視窗類呢?create函式第乙個引數(其它引數請參考msdn或《深出淺出mfc》詳解)指定視窗類設為null又是什麼意思啊?意思是要以mfc內建的空中類產生乙個標準的外框視窗,但,我們的程式一般都沒有註冊任何視窗類呀!噢,create函式在產生視窗之前會引發視窗類的註冊操作。

讓我們先挖出create函式都做了些什麼操作,create函式定義於winfrm.cpp的第538行(在此我就不把**copy過來了,你自己開啟出來看吧),函式在562行呼叫createex函式,由於createex是cwnd的成員函式,而cframewnd是從cwnd繼而來,故將呼叫cwnd::createex。此函式定義於wincore.cpp第665行,下面是部分**:

bool cwnd::createex(dword dwexstyle, lpctstr lpszclassname, 

lpctstr lpszwindowname, dword dwstyle,

int x, int y, int nwidth, int nheight,

hwnd hwndparent, hmenu nidorhmenu, lpvoid lpparam)

afxhookwindowcreate(this);

hwnd hwnd = ::createwindowex(cs.dwexstyle, cs.lpszclass,

cs.lpszname, cs.style, cs.x, cs.y, cs.cx, cs.cy,

cs.hwndparent, cs.hmenu, cs.hinstance, cs.lpcreateparams);

...}

用過sdk程式設計序的朋友,看到上面**應該有一點感覺了吧,函式中呼叫的precreatewindows是虛函式,在cwnd和cframewnd之中都有定義。由於this指標所指物件的緣故,這裡應該呼叫的是cframewnd::precreatewindow。該函式定義於winfrm.cpp第521行,以下是部分**:

bool cframewnd::precreatewindow(createstruct& cs)

...}

其中afxdeferregisterclass是乙個定義於afximpl.h中的巨集。該巨集如下:

#define afxdeferregisterclass(fclass) afxenddeferregisterclass(fclass)

注:這裡有巨集和《深入淺出mfc》的不一樣,以上**是從visual c++ 6.0摘取。

afxenddeferregisterclass定義於wincore.cpp第3619行,該函式很複雜,主要是註冊五個視窗類(哇!終於看到視窗類了,怎麼用5個呢?我還不清楚),不同類的precreatewindow成員函式都是在視窗產生之前一刻被呼叫,準備用來註冊視窗類。如果我們指定的視窗類是null,那麼就使用系統預設類。從cwnd及其各個派生類的precreatewindow成員函式可以看出,整個framework針對不同功能的視窗使用了哪些視窗類。

7、視窗顯示與更新

函式呼叫cwinthread::run函式,該函式定義於thrdcore.cpp第456行,在這裡我就不copy出來了。函式在第480行呼叫了pumpmessage函式,該函式定義於thrdcore.cpp第810行,整理後的部分**如下:

bool cwinthread::pumpmessage()

// process this message

if (m_msgcur.message != wm_kickidle && !pretranslatemessage(&m_msgcur))

return true;

}該函式主要操作是將訊息由::dispatchmessage送到視窗函式(cwnd::defwindowproc)中,但程式一般沒有提供任何視窗函式,但在afxenddeferregisterclass中,在註冊五種視窗類之前已經指定視窗函式為:

wndcls.lpfnwndproc = defwindowproc;

雖然視窗函式被指定為defwindowproc成員函式(cwnd::defwindowproc),但事實上訊息並不是被唧往該處,而是乙個名為afxwndproc的全域性函式去。

9、把訊息與處理函式連線在一起——message map機制

到此,主視窗已經產生,等待的就是各種訊息了,然後呼叫相應的處理函式,然而訊息和處理函式怎樣連線在一起呢?mfc採用了message map機制(訊息對映機制),提供給應用程式使用的「很方便的介面」的兩組巨集,其原理我還不大清楚,在這裡也無法講解,主要用法是:先在類宣告中結合declare_message_map()給出處理函式,如:

class cmyframewnd : public cframewnd

再在相應的.cpp檔案的任何位置(當然不能在函式之內)使用bebin_message_map()和end_message_map()巨集把相應的訊息加入去,如:

begin_message_map(cmyframewnd, cframewnd)

on_command(idm_about, onabout)

on_wm_paint()

end_message_map()

總結一下

(2) 呼叫winmain函式完成初始化工作: 通過巨集_twinmain

(3) 初始化工作包括: 視窗類註冊、視窗產生、顯示和更新、訊息迴圈等等

① 註冊視窗類:afxenddeferregisterclass()  //相當於sdk裡面的registerclass()函式

② 建立視窗:cmainframe::precreatewindow()  //允許我們修改視窗屬性的地方

cframewnd::create()

③ 訊息迴圈:pumpmessage()

自動對焦 AF

af效能是判斷相機好壞的重要指標,主要從準確度和速度兩個方面來進行考察,本文將介紹自動對焦的幾種方式。自動對焦目前主流的大致上有cdaf,pdaf和laser assist af這三種。前兩種屬於被動式,也就是說相機不需要其他額外輔助的探測手段,直接使用入射光線做判斷來對焦 而第三種屬於主動式,也就...

SpringMVC框架流程

springmvc流程圖 1 dispatcherservlet 前端控制器 控制器 mvc的核心 在web.xml中配置,dispatcherservlet是整個流程控制的核心,由它呼叫其他元件處理使用者的請求,降低了元件之間的耦合度。作用 接收請求,響應結果,相當於 器,處理器,減少其他元件之間...

AF監聽網路改變

可以使用afn框架中的afnetworkreachabilitymanager來監聽網路狀態的改變,也可以利用蘋果提供的reachability來監聽。建議在開發中直接使用afn框架處理。示例 如下 void afn manger startmonitoring reachablity 是乙個ios...