MFC的MVC

摘錄了幾篇文章:

MFC應用程序中指針的使用
作者:gouguijia

1) 在View中獲得Doc指針
2) 在App中獲得MainFrame指針
3) 在View中獲得MainFrame指針
4) 獲得View(已建立)指針
5) 獲得當前文檔指針
6) 獲得狀態欄與工具欄指針
7) 獲得狀態欄與工具欄變量
8) 在Mainframe獲得菜單指針
9) 在任何類中獲得應用程序類
10) 從文檔類取得視圖類的指針(1)
11) 在App中獲得文檔模板指針
12) 從文檔模板獲得文檔類指針
13) 在文檔類中獲得文檔模板指針
14) 從文檔類取得視圖類的指針(2)
15) 從一個視圖類取得另一視圖類的指針

VC中編程對於剛剛開始學習的同學,最大的障礙和問題就是消息機制和指針獲取與操作。其實這些內容基本上是每本VC學習工具書上必講的內容,而且通過MSDN很多問題都能解決。
下面文字主要是個人在編程中指針使用的一些體會,說的不當的地方請指正。
一般我們使用的框架是VC提供的Wizard生成的MFC App Wizard(exe)框架,無論是多文檔還是單文檔,都存在指針獲取和操作問題。
下面這節內容主要是一般的框架,然後再講多線程中的指針使用。使用到的類需要包含響應的頭文件。首先一般獲得本類(視,文檔,對話框都支持)實例指針 this,用this的目的,主要可以通過類中的函數向其他類或者函數中髮指針,以便於在非本類中操作和使用本類中的功能。

1) 在View中獲得Doc指針
CYouSDIDoc *pDoc=GetDocument();一個視只能有一個文檔。
2) 在App中獲得MainFrame指針
CWinApp 中的 m_pMainWnd變量就是MainFrame的指針
也可以:
CMainFrame *pMain =(CMainFrame *)AfxGetMainWnd();
3) 在View中獲得MainFrame指針
CMainFrame *pMain=(CmaimFrame *)AfxGetApp()->m_pMainWnd;
4) 獲得View(已建立)指針
CMainFrame *pMain=(CmaimFrame *)AfxGetApp()->m_pMainWnd;
CyouView *pView=(CyouView *)pMain->GetActiveView();
5) 獲得當前文檔指針
CDocument * pCurrentDoc =(CFrameWnd *)m_pMainWnd->GetActiveDocument();
6) 獲得狀態欄與工具欄指針
CStatusBar * pStatusBar=(CStatusBar *)AfxGetMainWnd()->GetDescendantWindow(AFX_IDW_STATUS_BAR);
CToolBar * pToolBar=(CtoolBar *)AfxGetMainWnd()->GetDescendantWindow(AFX_IDW_TOOLBAR);
7) 如果框架中加入工具欄和狀態欄變量還可以這樣
(CMainFrame *)GetParent()->m_wndToolBar;
(CMainFrame *)GetParent()->m_wndStatusBar;
8) 在Mainframe獲得菜單指針
CMenu *pMenu=m_pMainWnd->GetMenu();
9) 在任何類中獲得應用程序類
用MFC全局函數AfxGetApp()獲得。

10) 從文檔類取得視圖類的指針
我是從http://download.cqcnc.com/soft/program/article/vc/vc405.html學到的,從文檔獲得視圖類指針目的一般爲了控制同一文檔的多個視圖的定位問題,我的體會特別是文字處理CEditView當產生多個視圖類時,這個功能是非常需要的。
CDocument類提供了兩個函數用於視圖類的定位:
GetFirstViewPosition()和GetNextView()
virtual POSITION GetFirstViewPosition() const;
virtual CView* GetNextView(POSITION& rPosition) const;

注意:GetNextView()括號中的參數用的是引用方式,因此執行後值可能改變。
GetFirstViewPosition()用於返回第一個視圖位置(返回的並非視圖類指針,而是一個POSITION類型值), GetNextView()有兩個功能:返回下一個視圖類的指針以及用引用調用的方式來改變傳入的POSITION類型參數的值。很明顯,在Test程序 中,只有一個視圖類,因此只需將這兩個函數調用一次即可得到CTestView的指針如下(需定義一個POSITION結構變量來輔助操作):
CTestView* pTestView;
POSITION pos=GetFirstViewPosition();
pTestView=GetNextView(pos);
這樣,便可到了CTestView類的指針pTestView.執行完幾句後,變量pos=NULL,因爲沒有下一個視圖類,自然也沒有下一個視圖類的 POSITION.但是這幾條語句太簡單,不具有太強的通用性和安全特徵;當象前面說的那樣,當要在多個視圖爲中返回某個指定類的指針時,我們需要遍歷所 有視圖類,直到找到指定類爲止。判斷一個類指針指向的是否某個類的實例時,可用IsKindOf()成員函數時行檢查,如:
pView->IsKindOf(RUNTIME_CLASS(CTestView));
即可檢查pView所指是否是CTestView類。

有了以上基礎,我們已經可以從文檔類取得任何類的指針。爲了方便,我們將其作爲一個文檔類的成員函數,它有一個參數,表示要獲得哪個類的指針。實現如下:
CView* CTestDoc::GetView(CRuntimeClass* pClass)
{
CView* pView;
POSITION pos=GetFirstViewPosition();

while(pos!=NULL){
pView=GetNextView(pos);
if(!pView->IsKindOf(pClass))
break;
}

if(!pView->IsKindOf(pClass)){
AfxMessageBox("Connt Locate the View./r/n http://www.VCKBASE.com");
return NULL;
}

return pView;
}

其中用了兩次視圖類的成員函數IsKindOf()來判斷,是因爲退出while循環有三種可能:

1.pos爲NULL,即已經不存在下一個視圖類供操作;
2.pView已符合要求。

1和2同是滿足。這是因爲GetNextView()的功能是將當前視圖指針改變成一個視圖的位置同時返回當前視圖指針,因此pos是pView的下一個 視圖類的POSITION,完全有可能既是pos==NULL又是pView符合需要。當所需的視圖是最後一個視圖是最後一個視圖類時就如引。因此需採用 兩次判斷。
使用該函數應遵循如下格式(以取得CTestView指針爲例):
CTestView* pTestView=(CTestView*)GetView(RUNTIME_CLASS(CTestView));
RUNTIME_CLASS是一個宏,可以簡單地理解它的作用:將類的名字轉化爲CRuntimeClass爲指針。
至於強制類型轉換也是爲了安全特性考慮的,因爲從同一個基類之間的指針類型是互相兼容的。這種強制類型轉換也許並不必要,但能避免一些可能出現的麻煩。

3.從一個視圖類取得另一視圖類的指針 綜合1和2,很容易得出視圖類之間互相獲得指針的方法:就是用文檔類作中轉,先用1的方法得到文檔類的指針,再用2的方法,以文檔類的視圖定位函數取得另一個視圖類。同樣,可以實現成一個函數:
(假設要從CTestAView中取得指向其它視圖類的指針)
CView* CTestAView::GetView(CRuntimeClass* pClass)
{
CTestDoc* pDoc=(CTestDoc*)GetDocument();
CView* pView;
POSITION pos=pDoc->GetFirstViewPosition();
while(pos!=NULL){
pView=pDoc->GetNextView(pos);
if(!pView->IsKindOf(pClass))
break;
}
if(!pView->IsKindOf(pClass)){
AfxMessageBox("Connt Locate the View.");
return NULL;
}

return pView;
}
這個函數和2中的GetView()相比,一是多了第一句以取得文檔類指針,二是在GetFirstViewPosition()和GetNextView()前加上了文檔類指針,以表示它們是文檔類成員函數。
有了此函數;當要從CTestAView中取得CTestBView的指針時,只需如下:
CTestBView* pTestbView=(CTestView*)GetView(RUNTIME_CLASS(CTestBView));
11)對於單文檔中也可以加入多個文檔模板,但是一般的開發就使用MDI方式開發多文檔模板,其方法與上述視圖的獲取方法很接近,這裏稍做解釋,如果不清楚,請查閱MSDN,(以下四個內容(11、12、13、14)來源:http://sanjianxia.myrice.com/vc/vc45.htm

可以用CWinApp::GetFirstDocTemplatePostion獲得應用程序註冊的第一個文檔模板的位置;
利用該值來調用CWinApp::GetNextDocTemplate函數,獲得第一個CDocTemplate對象指針。
POSITION GetFirstDocTemplate( ) const; 
CDocTemplate *GetNextDocTemplate( POSITION & pos ) const;
第二個函數返回由pos 標識的文檔模板。POSITION是MFC定義的一個用於迭代或對象指針檢索的值。通過這兩個函數,應用程序可以遍歷整個文檔模板列表。如果被檢索的文檔模板是模板列表中的最後一個,則pos參數被置爲NULL。

12)一個文檔模板可以有多個文檔,每個文檔模板都保留並維護了一個所有對應文檔的指針列表。
用CDocTemplate::GetFirstDocPosition函數獲得與文檔模板相關的文檔集合中第一個文檔的位置,並用POSITION值作爲CDocTemplate::GetNextDoc的參數來重複遍歷與模板相關的文檔列表。函數原形爲:
viaual POSITION GetFirstDocPosition( ) const = 0; 
visual CDocument *GetNextDoc(POSITION & rPos) const = 0;
如果列表爲空,則rPos被置爲NULL.

13)在文檔中可以調用CDocument::GetDocTemplate獲得指向該文檔模板的指針。函數原形如下:
CDocTemplate * GetDocTemplate ( ) const; 
如果該文檔不屬於文檔模板管理,則返回值爲NULL。

14)一個文檔可以有多個視。每一個文檔都保留並維護一個所有相關視的列表。CDocument::AddView將一個視連接到文檔上,將該視加入到文 檔相聯繫的視的列表中,並將視的文檔指針指向該文檔。當有File/New、File/Open、Windows/New或Window/Split的命 令而將一個新創建的視的對象連接到文檔上時, MFC會自動調用該函數,框架通過文檔/視的結構將文檔和視聯繫起來。當然,程序員也可以根據自己的需要調用該函數。
Virtual POSITION GetFirstViewPosition( ) const; 
Virtual CView * GetNextView( POSITION &rPosition) cosnt;
應用程序可以調用CDocument::GetFirstViewPosition返回與調用文檔相聯繫的視的列表中的第一個視的位置,並調用 CDocument::GetNextView返回指定位置的視,並將rPositon的值置爲列表中下一個視的POSITION值。如果找到的視爲列表 中的最後一個視,則將rPosition置爲NULL.

15)從一個視圖類取得另一視圖類的指針
這個應用在多視的應用程序中很多見,一般如果自己在主程序或者主框架中做好變量記號,也可以獲得,還有比較通用的就是用文檔類作中轉,以文檔類的視圖遍歷定位,取得另一個視圖類。這個功能從本文第10項中可以得到。

這些資料大部分都是從網上和MSDN中獲得的,寫這個文檔就是爲了讓大家不用再尋找,列出標題,可操作性更強。

基於MFC文檔/視/框架程序之利劍

在我的同事當中,有不少人在用VC/MFC,但許多人總是容易被MFC的文檔視這種框架弄得暈頭轉向,儘管在他們的桌子旁堆了很多有關VC的資料。在我的學習過程中我發現下面的理解以及一些調用函數是最爲關鍵的,每當有人問我有關與此的問題,我給他(她)共享了下面的文檔後就不用重複解釋很多東西了。儘管這些都能從MSDN中最終找到答案,但有些人就是懼怕MSDN;希望您擁有這份資料後,可以大大減少你的懼怕。

l         預備

1.         文檔類通常與數據的存取以及其它處理過程有關,而視主要用來表現文檔數據的,但每個視必須依附於一個框架(在SDI中依附於MainFrame,在MDI中依附於ChildFrame),框架上可以放置工具欄、菜單等;

2.         某種類型的文檔、框架、視共同關聯構成一個文檔模板類(基類是CDocTemplate,其子類CSingleDocTemplateCMutiDocTemplate分對應於SDIMDI),文檔模板類在WinApp的初始化時構造;

3.         一個APP可以有N個文檔模板,但SDIMDI框架嚮導默認生成的都只有一種模板(若需要更多則可以模仿添加);一個文檔模板在程序中可多次生成多個文檔實例,此所謂多文檔接口MDI(多文檔不僅可以是多個同類型的文檔還可以是多個不同類型的文檔);一種數據可以有N種表現形式,所以一個文檔可以有N個視與之關聯;

4.         在某一時刻只可能有一個活動的文檔、視、和框架。即當前文檔、視、和框架。

理解了這些再加上下面的操作就可以令你處理這類問題時得心應手了。

l         MDI對象之間的交互函數

函數

返回值

Global

AfxGetApp

CWinApp *(指向當前程序)

Global

AfxGetMainWnd

CWnd *(指向程序主窗口框架)

CMDIFrameWnd

MDIGetActive

CMDIChild Wnd *(指向當前活動的MDI子窗口)

CWnd

GetParentFrame

CFrameWnd *(指向父窗口框架)

CFrameWnd

GetActiveView

CView*(指向當前活動視)

CFrameWnd

GetActiveDocument

CDocument *(指向當前活動文檔)

CView

GetDocument

CDocument* (同該視圖類相聯繫)

CDocument

GetFirstViewPosition

同文檔相關的視圖列表中的第一個視圖類位置

CDocument

GetNextView

同文檔相關的視圖列表中的下一個視圖類位置

e.g.        CMainFrame* mfrm=(CMainFrame *)AfxGetMainWnd();

e.g.:得到當前視:

        CFrameWnd* pMain = (CFrameWnd*) AfxGetMainWnd();

       CFrameWnd *pChild = (CFrameWnd *) pMain->GetActiveFrame();

       CView* m_pAgentView = (CView *)pChild->GetActiveView();

l         訪問當前活動視圖和活動文檔

1.         對於SDI程序,主框架窗口就是文檔框窗,可以採用以下方法取得當前文檔和視圖:

u       取得活動文檔:

CMyDocument* pDoc;

pDoc=(CMyDocument*)((CFrameWnd*)AfxGetApp()->m_pMainWnd)->GetActiveDocument();

pDoc=(CMyDocument*)((CFrameWnd*)AfxGetMainWnd())

這兩者是等效的。

u       取得活動視圖:

CMyView* pView;

pView=(CMyView*)((CFrameWnd*)AfxGetApp()->m_pMainWnd)->GetActiveView();

2.  對於MDI程序,由於子窗口才是文檔框窗,因此首先要用GetActiveFrame()取得活動子框架窗口,然後通過該子窗口獲取活動文檔和視圖:

CMDIChildWnd* pChild=(CMDIChildWnd*)((CFrameWnd*)AfxGetApp()->m_pMainWnd)->GetActiveFrame();

u     取得活動文檔:

CMyDocument* pDoc=pChild->GetActiveDocument();

CMyView* pView=(CMyView*)pChild->GetActiveView();

可以把上述函數片段做成靜態成員函數,比如:

static CMyDocument::GetCurrentDoc()

{

CMDIChildWnd* pChild=(CMDIChildWnd*)((CFrameWnd*)AfxGetApp()- >m_pMainWnd)->GetActiveFrame();

CMyDocument* pDoc=pChild->GetActiveDocument();

}

這樣就可以通過以下方式取得活動文檔(或視圖):

CMyDocument::GetCurrentDoc();

注:靜態成員函數調用時不需要一個具體的對象與之相關聯。

l         文檔模板中找文檔

例:

       CMultiDocTemplate* pDocTmpl;

       pDocTmpl  = ((CImpApp*)AfxGetApp())->m_pDocTemplate;

       POSITION posDoc= pDocTmpl->GetFirstDocPosition();

       while(NULL!=posDoc)

       {

              pFrmDoc = (CFrmDoc*) pDocTmpl->GetNextDoc(posDoc);

              nDocOrder++;

       }

願這些能成爲您操作MDISDI類程序的利劍,歡迎補充和提意見!

Trackback: http://tb.blog.csdn.net/TrackBack.aspx?PostId=16153

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章