• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            VC 和 MFC 的一些常見(jiàn)問(wèn)題

            如何拋出(throw)由CUserException派生的異常?

            當(dāng)我試圖捕獲(catch)一個(gè)派生類(lèi)異常時(shí),我得到以下錯(cuò)誤"error C2039:'classCMyException': is not a member of 'CMyException' 'classCMyException': undeclared identifier 'IsKindOf': cannot convert parameter 1 from 'int*' to 'const struct CRuntimeClass*"

            你必需通過(guò)使用DECLARE_DYNAMIC()和IMPLEMENT_DYNAMIC()宏來(lái)使你的CMyException類(lèi)可以動(dòng)態(tài)地創(chuàng)建。CATCH宏希望能夠得到關(guān)于被拋出類(lèi)的運(yùn)行時(shí)刻信息。

            異常類(lèi)一定要從CUserException中派生出來(lái)嗎?

            不,CUserException中的"User"僅僅指用戶產(chǎn)生的異常。而把它當(dāng)作你所能派生的唯一異常是種常見(jiàn)的誤解。

            如何從HDC建立一個(gè)CDC類(lèi)?

            有時(shí)Windows API將會(huì)給你一個(gè)DC句柄,你可以通過(guò)它建立一個(gè)CDC類(lèi)。例如:下拉式列表、組合框和按鈕。通過(guò)hDC你將接收到繪制消息。下面是將HDC轉(zhuǎn)換成你更熟悉的CDC的程序段。你也可以將該技巧用在其他任何MFC類(lèi)和Windows句柄的轉(zhuǎn)換中。

            void MyODList::DrawItem(LPDRAWITEMSTRUCT lpDrawItem)
            {
            CDC myDC;
            myDC.Attach(lpDrawItem->hDC);
            //在此插入其他需要的代碼。
            //如果你不將句柄分離,它將被刪除,從而導(dǎo)致問(wèn)題。
            myDC.Detach();
            }
            
            另一個(gè)方法是調(diào)用CDC類(lèi)的FromHandle方法:
            CDC * pDC = CDC:FromHandle(lpDrawItem->hDC);
            
            目前還不清楚哪種方法更優(yōu)越―使用FromHandle()的錯(cuò)誤也許會(huì)更少些,因?yàn)樗灰竽惴蛛x(detach)句柄。

            如何從磁盤(pán)上讀取256色位圖文件?

            當(dāng)前,MFC并不支持直接讀取和顯示DIB文件和BMP文件。然而,有很多樣例應(yīng)用程序能夠說(shuō)明如何完成該項(xiàng)任務(wù)。第一個(gè)例子是MFC樣例程序DIBLOOK。樣例MULTDOCS用DIBLOOK提供的相同源代碼來(lái)讀取并顯示DIB文件和BMP文件。其他兩個(gè)VC++中附帶的例子是SDK軟件包中的DIBVIEW程序和SHOWDIB程序。

            如何改變一個(gè)視圖的大小?

            通常,你可以調(diào)用函數(shù)MoveWindow()來(lái)改變窗口的大小。在用MFC庫(kù)開(kāi)發(fā)的應(yīng)用程序中, 視圖是被框架窗口所圍繞的一個(gè)子窗口。為了改變一個(gè)視圖的大小,你可以通過(guò)調(diào)用函數(shù)GetParentFrame()來(lái)得到框架窗口的指針,然后調(diào)用函數(shù)MoveWindow()來(lái)改變父窗口的大小。當(dāng)父框架窗口改變大小時(shí),視圖也會(huì)自動(dòng)地改變大小來(lái)適應(yīng)父窗口。

            如何改變一個(gè)CFormView的大小?

            要想詳細(xì)了解的話,你可以看有關(guān)Visual C++基礎(chǔ)知識(shí)的文章Q98598 《Using CFormView in SDI and MDI Applications》。基本上,在從CFormView類(lèi)派生出來(lái)的類(lèi)中,你必須覆蓋函數(shù)OnInitialUpdate()。其他有關(guān)建立CFormView的細(xì)節(jié)問(wèn)題,可以從該文章中獲得。

            在類(lèi)ClikethisView中聲明如下函數(shù):
            virtual void OnInitialUpdate();
            在ClikethisView的代碼中,函數(shù)如下:
            void ClikethisView::OnInitialUpdate()
            {
            //使窗口與主對(duì)話框同樣大小
            CFormView::OnInitialUpdate();
            GetParentFrame()->RecalcLayout();
            ResizeParentToFit( /*FALSE*/ );
            }
            

            如何使用一個(gè)文檔模板的新視圖?

            在用AppWizard創(chuàng)建的應(yīng)用程序中,你有兩種選擇:改變當(dāng)前視圖的派生關(guān)系或者建立一個(gè)新視圖并且在你的MDI程序中同時(shí)利用新視圖和原先的視圖。


            為了創(chuàng)建一個(gè)新視圖,你可以用ClassWizard由CView派生一個(gè)新的類(lèi)。當(dāng)新類(lèi)創(chuàng)建以后,利用新視圖或修改由AppWizard提供的視圖,兩者的步驟是相同的。


            修改視類(lèi)的頭文件,從而將所有對(duì)CView類(lèi)的引用改名為你所想要的名稱(chēng)。本例中的類(lèi)由CScrollView派生而來(lái)。通常,這個(gè)步驟包括對(duì)類(lèi)的改變,視類(lèi)將由如下方式派生而來(lái):
                class CMyView : public CScrollView


            修改視類(lèi)的實(shí)現(xiàn)文件,從而將所有對(duì)CView的引用改名為你所想要的名稱(chēng)。這包括將IMPLEMENT_DYNCREATE那一行的語(yǔ)句改為:
                IMPLEMENT_DYNCREATE(CMyView, CScrollView)


            將BEGIN_MESSAGE_MAP那一行的語(yǔ)句改為:
                BEGIN_MESSAGE_MAP(CMyView, CScrollView)


            并且將其他所有的CView改成CScrollView.


            假如你修改的視圖是由AppWizard生成的,那么就不需要作更多的修改了。而如果你在創(chuàng)建一個(gè)新視圖,先在CWinApp::InitInstance()函數(shù)中找到對(duì)AddDocTemplate()函數(shù)的調(diào)用。AddDocTemplate()函數(shù)的第三個(gè)參數(shù)是RUNTIME_CLASS(CSomeView),用CMyView來(lái)代替CSomeView,就可以將當(dāng)前視圖改為新視圖。在MDI應(yīng)用程序中,你可以增加第二個(gè)AddDocTemplate()函數(shù)調(diào)用來(lái)使用多視圖類(lèi)型,將RUNTIME_CLASS(CSomeView)改為RUNTIME_CLASS (CMyView)。

            要想獲得更多的信息請(qǐng)參閱Q99562中相關(guān)文章《Switching Views in a Single Document Interface Program》 。

            如何改變視圖的背景色?

            你可以通過(guò)處理WM_ERASEBKGND消息來(lái)改變CView、CFrameWnd或CWnd對(duì)象的背景色。請(qǐng)看如下的程序段:

              BOOL CSampleView::OnEraseBkgnd(CDC* pDC)
            {
            // 設(shè)置所要求背景色的刷子
            CBrush backBrush(RGB(255, 128, 128));
            // 保存舊刷子
            CBrush* pOldBrush = pDC->SelectObject(&backBrush);
            CRect rect;
            pDC->GetClipBox(&rect);     // 擦除所需的區(qū)域
            pDC->PatBlt(rect.left, rect.top, rect.Width(), rect.Height(), PATCOPY);
            pDC->SelectObject(pOldBrush);
            return TRUE;
            }
            
            而我則用如下方法解決這個(gè)問(wèn)題:
              HBRUSH dlgtest::OnCtlColor(CDC* pDC, CWnd* pWnd, UINT nCtlColor)
            {
            switch (nCtlColor)
            {
            case CTLCOLOR_BTN:
            case CTLCOLOR_STATIC:
            {
            pDC->SetBkMode(TRANSPARENT);
            }
            case CTLCOLOR_DLG:
            {
            CBrush*     back_brush;
            COLORREF    color;
            color = (COLORREF) GetSysColor(COLOR_BTNFACE);
            back_brush = new CBrush(color);
            return (HBRUSH) (back_brush->m_hObject);
            }
            }
            return(CFormView::OnCtlColor(pDC, pWnd, nCtlColor));
            }
            

            如何得到當(dāng)前視圖?

            最佳方法是將視圖當(dāng)作一個(gè)參數(shù)來(lái)傳遞。如果不能這樣做,但你確信它是當(dāng)前激活文檔和當(dāng)前激活視圖的話,你也可以得到該視圖。具體細(xì)節(jié)見(jiàn)Visual C++文章Q108587《Get Current CDocument or CView from Anywhere》。

            簡(jiǎn)單說(shuō)來(lái),用:
            ((CFrameWnd*) AfxGetApp()->m_pMainWnd))->GetActiveDocument()
            和:
            ((CFrameWnd*)(AfxGetApp()->m_pMainWnd))->GetActiveView()
            
            來(lái)得到文檔和視圖。一個(gè)好的方法是將它們封裝在你的CMyDoc和CMyView類(lèi)的靜態(tài)函數(shù)中,并且核對(duì)它們是否屬于正確的RUNTIME_CLASS。然而,假如這個(gè)視圖不是當(dāng)前激活視圖或者你在運(yùn)行OLE本地激活,這樣將不成功。

            如何在一個(gè)文檔中建立多個(gè)視圖?

            CDocTemplate::CreateNewFrame()函數(shù)創(chuàng)建MFC MDI應(yīng)用程序中的文檔的附加視圖。為了調(diào)用該函數(shù),要指定一個(gè)指向CDocument對(duì)象(指將為之建立視圖的文檔)的指針和一個(gè)指向可從中復(fù)制屬性的框架窗口的指針。一般情形下,該函數(shù)的第二個(gè)參數(shù)為NULL。

            當(dāng)應(yīng)用程序調(diào)用函數(shù)CreateNewFrame()時(shí),該函數(shù)就創(chuàng)建一個(gè)框架窗口和在該窗口內(nèi)的視圖。框架窗口和它的視圖的類(lèi)型由與CreateNewFrame()函數(shù)調(diào)用指定的文檔相關(guān)的文檔摸板(CDocTemplate)決定。

            Visual C++中的CHKBOOK MFC樣例程序也演示了如何為文檔建立附加的框架和視圖。檢查CHKBOOK.CPP文件中的CChkBookApp::OpenDocumentfile()函數(shù)。

            另一個(gè)用函數(shù)CreateNewFrame()的例子是MULTVIEW樣本程序。

            CreateNewFrame()函數(shù)建立了一個(gè)框架和一個(gè)視圖,而不僅僅是一個(gè)視圖。假如CreateNewFrame()函數(shù)不能完全符合你的需要,可參考CreateNewFrame()函數(shù)的源程序來(lái)了解對(duì)建立結(jié)構(gòu)和視圖所必須的步驟。

            如何在MDI程序中得到所有的視圖?

            你必須用一些文檔中沒(méi)有記載的函數(shù):

              CDocument::GetFirstViewPosition(); // DOCCORE.CPP
            CDocument::GetNextView(); // DOCCORE.CPP
            CMultiDocTemplate::GetFirstDocPosition(); // DOCMULTI.CPP
            CMultiDocTemplate::GetNextDoc(); // DOCMULTI.CPP
            

            你還需要與CWinApp的成員m_templateList打交道。
            注意:在MFC 版本4.0中已改變。現(xiàn)在已經(jīng)有一個(gè)叫CDocManager的類(lèi)可以幫助你顯示所有的視圖和文檔。請(qǐng)參考《MFC Internals》獲得更詳細(xì)的信息。

            如何建立一個(gè)可用鼠標(biāo)拉動(dòng)的CScrollView類(lèi)

            在CIS上從MSMFC庫(kù)下載AUTOSV.LZH。這個(gè)程序告訴你如何實(shí)現(xiàn)一個(gè)輔助消息循環(huán)來(lái)管理鼠標(biāo)的活動(dòng),并提供了鉤掛來(lái)對(duì)代碼進(jìn)行定制。這是一個(gè)免費(fèi)軟件。

            一定要用視圖/文檔結(jié)構(gòu)嗎?

            MFC并不一定要求你使用文檔/視圖結(jié)構(gòu)。查看HELLO、 MDI和HELLOAPP例子―它們就沒(méi)有用那種結(jié)構(gòu)。大多數(shù)MFC特性都可以在非文檔/視圖應(yīng)用程序中得到運(yùn)用。但是當(dāng)你不用文檔 / 視圖結(jié)構(gòu)時(shí),你確實(shí)會(huì)失去一些特性,例如打印預(yù)覽和許多OLE特性。

            如何得到當(dāng)前文檔?

            請(qǐng)?jiān)敿?xì)參閱"如何得到當(dāng)前視圖?"章節(jié)。

            文檔何時(shí)被析構(gòu)?

            在SDI程序中,程序退出后文檔就被刪除。在MDI程序中,與該文檔相關(guān)的最后一個(gè)視圖關(guān)閉時(shí)文檔就被刪除。為了在SDI和MDI中同時(shí)用這個(gè)文檔,你應(yīng)該在虛函數(shù)DeleteContents()函數(shù)中刪除該文檔的數(shù)據(jù),而不是在析構(gòu)器中。

            如何建立多文檔?

            為了加入對(duì)附加文檔類(lèi)型的支持,你可以在CWinApp派生類(lèi)中創(chuàng)建和注冊(cè)附加CmultiDocTemplate對(duì)象。這種方法已經(jīng)在MULTDOCS樣例程序中得以說(shuō)明。將一個(gè)附加文檔類(lèi)型加入到MFC程序的一般步驟如下:

            用AppWizard來(lái)創(chuàng)建一個(gè)新的文檔類(lèi)和視圖類(lèi)。
            用資源編輯器增加新的資源字串來(lái)支持新的文檔類(lèi)。要想知道關(guān)于文檔樣板字符串格式的更多內(nèi)容,請(qǐng)參閱"如何理解文檔樣板字符串"。

            用資源編輯器增加附加的應(yīng)用程序圖標(biāo)和菜單資源。注意,這些資源中每一個(gè)的ID都必須與在步驟2中創(chuàng)建的文檔模板字符串的ID是相同的。這個(gè)ID被CmultiDocTemplate類(lèi)用來(lái)識(shí)別與附加文檔類(lèi)型相關(guān)的資源。

            在應(yīng)用程序的InitInstance()函數(shù)中,創(chuàng)建了另一個(gè)CMultiDocTemplate對(duì)象并且用CWinApp::AddDocTemplate()函數(shù)來(lái)注冊(cè)。例如:

            CMultiDocTemplate* pDocTemplate2 = new CMultiDocTemplate(
            IDR_DOC2TYPE, RUNTIME_CLASS(CDoc2),
            RUNTIME_CLASS(CMDIChildWnd),RUNTIME_CLASS(CView2));
            AddDocTemplate(pDocTemplate2);
            
            最后,將定制的序列化和繪圖代碼加入到你的新文檔和視圖類(lèi)中。

            如何得到一個(gè)打開(kāi)文檔的列表?

            下面的程序段指明如何得到用CDocTemplate對(duì)象建立的所有文檔的指針列表。
            下面的程序段中,CMyApp由CWinApp派生而來(lái)。變量m_templateList是一個(gè)CPtrList對(duì)象,它是CwinApp的成員變量,包含一個(gè)所有文檔模板指針的列表。文檔模板函數(shù)GetFirstDocPosition()和GetNextDoc()被用來(lái)在文檔模板列表中進(jìn)行迭代來(lái)得到每一個(gè)文檔模板。

              void CMyApp::GetDocumentList(CObList * pDocList)
            {
            ASSERT(pDocList->IsEmpty());
            POSITION pos = m_templateList.GetHeadPosition();
            while (pos)
            {
            CDocTemplate* pTemplate =
            (CDocTemplate*)m_templateList.GetNext(pos);
            POSITION pos2 = pTemplate->GetFirstDocPosition();
            while (pos2)
            {
            CDocument * pDocument;
            if ((pDocument=pTemplate->GetNextDoc(pos2)) != NULL)
            pDocList->AddHead(pDocument);
            }
            }
            }
            

            在參考手冊(cè)或在線幫助中,有兩個(gè)CdocTemplate類(lèi)的公共成員函數(shù)沒(méi)有被說(shuō)明。然而, 這些公共成員函數(shù)在CDocTemplate類(lèi)中被定義,并且為在打開(kāi)文檔的列表中前后搜索提供了簡(jiǎn)單的支持。

            這些函數(shù)如下:


            Function virtual POSITION GetFirstDocPosition() const;
            調(diào)用該函數(shù)得到在打開(kāi)的文檔列表中與模板相關(guān)聯(lián)的第一個(gè)文檔的位置。返回的POSITION的值能夠被GetNextDoc成員函數(shù)反復(fù)使用。

            Function Virtual CDocument* GetNextDoc(POSITION& rPosition) const;
            rPostion是前面調(diào)用GetNextDoc 或GetFirstDocPosition成員函數(shù)返回的POSITION值。這個(gè)值不能是NULL。調(diào)用該函數(shù)來(lái)在所有打開(kāi)的文檔中進(jìn)行迭代。該函數(shù)返回被rPosition所標(biāo)識(shí)的文檔并將rPosition設(shè)置為列表中的下一個(gè)文檔的POSITION值。假如所檢索的是列表中的最后一個(gè)文檔,rPosition將被設(shè)為空值。

            注意,這僅對(duì)MFC3.2版本或更低版本有效,對(duì)MFC4.0版本請(qǐng)參考下面:

              void CMyApp::DoSomethingToAllDocs()
            {
            CObList  pDocList;
            POSITION pos = GetFirstDocTemplatePosition();
            while(pos)
            {
            CDocTemplate* pTemplate = GetNextDocTemplate(pos);
            POSITION pos2 = pTemplate->GetFirstDocPosition();
            while(pos2)
            {
            CDocument* pDocument;
            if(pDocument = pTemplate->GetNextDoc(pos2))
            pDocList.AddHead(pDocument);
            }
            }
            if(!pDocList.IsEmpty()){
            pos = pDocList.GetHeadPosition();
            while(pos)
            {
            //為每一個(gè)文檔調(diào)用CDocument函數(shù)
            ( (CDocument*)pDocList.GetNext(pos) )
            ->UpdateAllViews(NULL);
            }
            }
            

            如何使我的程序在啟動(dòng)時(shí)不創(chuàng)建一個(gè)新文檔?

            在程序的InitInstance中的ProcessShellCommand函數(shù)之前加入: cmdInfo.m_nShellCommand = CCommandLineInfo::FileNothing

            posted on 2008-12-26 16:45 wrh 閱讀(223) 評(píng)論(0)  編輯 收藏 引用


            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問(wèn)   Chat2DB   管理


            導(dǎo)航

            <2010年11月>
            31123456
            78910111213
            14151617181920
            21222324252627
            2829301234
            567891011

            統(tǒng)計(jì)

            常用鏈接

            留言簿(19)

            隨筆檔案

            文章檔案

            收藏夾

            搜索

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            亚洲AV乱码久久精品蜜桃| 久久青青国产| 亚洲国产欧美国产综合久久| 精品国产青草久久久久福利| 国产亚洲欧美成人久久片| 热re99久久精品国99热| 午夜人妻久久久久久久久| 国内精品伊人久久久影院| 欧美成人免费观看久久| 精品久久久久久久国产潘金莲| 久久久久久国产精品美女| 久久免费大片| 久久久噜噜噜久久中文字幕色伊伊 | 麻豆精品久久精品色综合| 久久久国产乱子伦精品作者| 午夜久久久久久禁播电影| 99久久婷婷国产综合亚洲| 热久久国产精品| 精品欧美一区二区三区久久久| 久久精品国产清自在天天线| 亚洲国产成人精品女人久久久 | 国产成人无码久久久精品一| 国产国产成人精品久久| 国产精品美女久久久久av爽| 久久精品国产色蜜蜜麻豆| 久久精品国产99国产精品导航| 久久久久亚洲AV无码麻豆| 99久久精品国产一区二区三区| 久久天天躁狠狠躁夜夜av浪潮| 久久精品国产AV一区二区三区| 久久香蕉国产线看观看乱码| 国产精品中文久久久久久久| 久久精品国产亚洲一区二区| 亚洲精品国精品久久99热| 久久精品九九亚洲精品| 久久久WWW成人免费精品| 久久久久久九九99精品| 久久一区二区三区99| 久久福利青草精品资源站| 久久国产AVJUST麻豆| 国产成人精品久久|