天天看點

MFC 中擷取各種類指針的方法

外部類中獲得doc

CMainFrame *pMain=(CMainFrame*)AfxGetApp()->m_pMainWnd; 

CDemoView *pView=(CDemoView *)pMain->GetActiveView();

CDemoDoc *pDoc=(CDemoDoc *)pView->GetDocument();

獲得各種視圖指針如下

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) 從一個視圖類取得另一視圖類的指針

16)擷取分割視圖中各個視圖的指針

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; or ((CMainFrame *)GetParent())->

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)從文檔獲得視圖類指針目的一般為了控制同一文檔的多個視圖的定位問題,我的體會特别是文字處理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 www.163.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的方法,以文檔類的視圖定位函數取得另 一個視圖類。同樣,可以實作成一個函數: (假設要從CTestA V i ew中取得指向其它視圖類的指針) CView* CTestA V i ew::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()前加上了文檔類指針,以表示它們是文檔類成員函數。 有了此函數;當要從CTestA V i ew中取得CTestBView的指針時,隻需如下:CTestBView* pTestbView=(CTestView*)GetView(RUNTIME_CLASS(CTestBView));

11)對于單文檔中也可以加入多個文檔模闆,但是一般的開發就使用MDI方式開發多文檔模闆,其方法與上述視圖的擷取方法很接近,這裡稍做解釋,如果不清楚,請查閱MSDN,

 可以用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項中可以得到。

16)擷取分割視圖中各個視圖的指針 CSplitterWnd m_wndSplitter; m_wndSplitter.CreateStatic(this, 1, 2);//分割成一行兩列 m_wndSplitter.CreateView(0, 0, RUNTIME_CLASS(CLeftPaneView), CSize(10, 10), pContext); m_wndSplitter.CreateView(0, 1, RUNTIME_CLASS(CRightPaneFrame), CSize(0, 0), pContext); //擷取左邊視圖的兩種方法 CLeftPaneView* pLeftPaneView = (CLeftPaneView*) m_wndSplitter.GetPane(0,0); //上一句可以用下句代替: //CLeftPaneView* pLeftPaneView = (CLeftPaneView *)GetActiveView(); //擷取右邊視圖 pLeftPaneView->m_pRightPaneFrame = (CRightPaneFrame*) m_wndSplitter.GetPane(0,1);