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)
可以用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中獲得的,寫這個文檔就是為了讓大家不用再尋找,列出標題,可操作性更強。