青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

Beginning to 編程

VC++ 方面編程文章

 

C++中實現串口操作類

工程下載:http://m.shnenglu.com/Files/richardzeng/C++中實現串口操作類%20SerialPortLib.rar

最近封裝了一個串口類,與大家分享,該類的主要特點是:能實現數據的異步接收;無須MFC的支持;只能在VS2003編譯通過,但只要做少量修改就可以在VC6.0中使用.使用起來非常簡單,主要代碼如下:

 1 #include "stdafx.h"
 2 #include "comm_exception.h"
 3 #include "SerialPort.h"
 4 #include "serialportobservertest.h"
 5 
 6 using namespace C2217::StdLib;
 7 using namespace
 IBMS;
 8 

 9 int _tmain(int argc, _TCHAR* argv[])
10 
{
11     try

12     {    
13         //聲明一個串口觀察者

14         CSerialPortObserverTest portObserver;
15         //聲明串口1

16         CSerialPort port(1);
17         //注冊串口的觀察者

18         port.AtachPortObserver(&portObserver);
19         //打開串口

20         port.Open();
21         byte data[100= {0
};
22 

23         port.Send(data,sizeof(data));
24 
    }
25     catch(comm_exception &
e)
26 
    {
27 
        SET_CATCH_POS(e);
28         std::cout <<
 e;
29 
    }
30 

31     return 0;
32 
}
33 

   串口數據的接收在

void CSerialPortObserverTest::OnSerialPortReceive(CSerialPort *pSerialPort, byte *pData, size_t nDataLen)
{
 cout 
<< pSerialPort->GetName().c_str() << "Received Data: "<<
endl;
 
 
for(size_t i=0; i< nDataLen ;++
i )
 {
  cout 
<< pData[i] << " "
 ;
 }

 cout 
<<
endl;
}

   完成,你也可以不使用觀察者,直接重寫void CSerialPort::OnReceiveData(byte *pData, size_t nDataLen)可以獲得更好的執行效率。去掉觀察者對象list.

   有什么問題郵件聯系:dyj057@gmail.com



# re: C++中實現串口操作類 2005-12-22 18:34 小明
我看你的程序使用了一個叫IbmsSerialPort.dll的dll來完成通訊

而這個IbmsSerialPort.dll首先使用CreateFile,然后使用GetCommState等等一系列communications resource function來完成端口通訊

ok,學到了一些東西

  回復 

# re: C++中實現串口操作類 2006-03-01 10:44 msn:a.zlp@163.com
CreateFile對串口操作是獨占的,其他的應用程序就不能打開,怎么實現觀察者的角色呢?想請教樓主!msn:a.zlp@163.com  回復
  

# re: C++中實現串口操作類 2006-03-01 12:00 天下無雙
這個簡單,當你發送數據的時候,也發送一份到觀察者.接收到數據的時候,也轉一份到觀察者。  回復 
   
  

posted @ 2006-03-10 11:15 Beginning to 編程 閱讀(1393) | 評論 (1)編輯 收藏

ACM學習網站

 
同濟大學的 Online Judge - http://acm.tongji.edu.cn/
浙江大學的 Online Judge - http://acm.zju.edu.cn/
北京大學的 Online Judge - http://acm.pku.edu.cn/
吉林大學的 Online Judge - http://acm.jlu.edu.cn/
四川大學的 Online Judge - http://cs.scu.edu.cn/acm
汕頭大學的 Online Judge - http://acm.stu.edu.cn/
中科大的 Online Judge - http://acm.ustc.edu.cn/index.php
哈工大的 Online Judge - http://acm.hit.edu.cn/acm.php
西班牙的 Universidad de Valladolid - http://acm.uva.es/
俄羅斯烏拉爾大學 - http://acm.timus.ru/

posted @ 2006-03-10 11:01 Beginning to 編程 閱讀(386) | 評論 (0)編輯 收藏

時間和日歷類的設計

     摘要: 時間和日歷類的設計(Java的Date和Calendar的C++實現)   C++通用框架的設計 作者:naven 1           介紹 時間和日歷以及時間的格式化處理在軟件的設計中起著非常重要的作用,但是目前C++的庫卻未有一個簡單易用的時間類,大部分都需要開發者直接調...  閱讀全文

posted @ 2006-03-10 10:56 Beginning to 編程 閱讀(653) | 評論 (0)編輯 收藏

結構化設計的救命稻草-回調機制 / 轉 Trackback: http://tb.blog.csdn.net/TrackBack.aspx?PostId=253623

摘要:開發模式的確立是軟件開發過程中不可缺少的一部分,就目前來說,面向過程和面向對象是兩種主要的設計方法,雖然面向對象OOP是比較流行的字眼,但不表示面向過程就一定好無作為,畢竟面向過程設計方法也有適合其應用的軟件系統:以功能操作為主,擴展性要求不高,無需過多考慮復用以及軟件的通用性能。那是不是面向過程的設計方法對于諸如系統框架擴展問題就絲毫沒有辦法了呢?

按照面向過程的基本原則,劃分系統功能模塊、模塊細分到函數、生成系統整體的結構模型,似乎在整個過程中沒有任何東西可以用來提供系統擴展,其實解決的方法還是有的,這根救命稻草就是回調機制。

一談到回調機制,當然就少不了我們的主角:系統API(通常都是)和回調函數,這兩者缺一不可。其實回調的基本思想就是由系統給我們提供一些接口,也就是常使用的API,這種函數可以將某個其他函數的地址作為其參數之一,而且可以利用該地址對這個函數進行調用,而被調用的函數就是我們通常所說的回調函數了。
下面給個回調函數使用的小例子:
------------------------------------------
//相當于我們提到的系統API
mainFunc( void*  userFunc )//當然參數不會這么簡單,只是模擬
{
 while (...)
 {
  printf("ok!");
                //調用回調函數了
  if (userFunc!=NULL) 
   userFunc();
 }
}
可以看出MainFunc可以根據函數userFunc的地址調用它。
------------------------------------------
這樣使用者只需要定義一個函數:void myFunc(),然后按照mainFunc(&myFunc)(&只表示傳遞的是函數的地址,無具體含義),就可以讓我們的mainFunc來調用myFunc從而實現相應的功能,這樣當然可以完成我們預期的目的-擴展現有系統。

在windows系統中,支持這種回調機制的系統API不占少數,像實現ListControl排序的SortItem()函數,還有操作Font使用的函數EnumFontFamilies()都有提供這種回調機制,使得我們的用戶有機會添加自己期望的功能實現。當然,使用回調函數并不是一個輕松的事情,如果我們的系統中存在了大量的回調函數是很難管理的,這個就與系統中存在大量全局變量一樣,出現多個函數爭相訪問同一個變量我們就很難使用簡單的邏輯來處理,容易陷入混亂,因此,盡管回調機制可以在某種程度上達到我們的目的,但切不可亂加使用,不然后果很難預料。

當然至于詳細的回調函數實現,還需要大家潛心研究,這里我只是總結一下:
1 回調函數是由開發者按照一定的原型進行定義的函數(每個回調函數都必須遵循這個原型來設計)

例如:
------------------------------------------
BOOL CALLBACK DialogProc(
    
     HWND hwndDlg, // handle of dialog box
     UINT uMsg, // message
     WPARAM wParam, // first message parameter
     LPARAM lParam // second message parameter
     );
------------------------------------------
說明:
回調函數必須有關鍵詞 CALLBACK
回調函數本身必須是全局函數或者靜態函數,不可定義為某個特定的類的成員函數

2 回調函數并不由開發者直接調用執行(只是使用系統接口API函數作為起點)
3 回調函數通常作為參數傳遞給系統API,由該API來調用
4 回調函數可能被系統API調用一次,也可能被循環調用多次(SortItem就是自調用)

最后說句題外話,其實windows系統中還有另一種機制-消息機制,也是一個比較不錯的工具,能夠為很多實際的問題提供解決方法,這個以后再總結了。

posted @ 2006-03-09 11:27 Beginning to 編程 閱讀(488) | 評論 (0)編輯 收藏

五大內存分區 /http://blog.csdn.net/welcome_ck/archive/2004/12/24/227961.aspx


    在C++中,內存分成5個區,他們分別是堆、棧、自由存儲區、全局/靜態存儲區和常量存儲區。
    棧,就是那些由編譯器在需要的時候分配,在不需要的時候自動清楚的變量的存儲區。里面的變量通常是局部變量、函數參數等。
    堆,就是那些由new分配的內存塊,他們的釋放編譯器不去管,由我們的應用程序去控制,一般一個new就要對應一個delete。如果程序員沒有釋放掉,那么在程序結束后,操作系統會自動回收。
    自由存儲區,就是那些由malloc等分配的內存塊,他和堆是十分相似的,不過它是用free來結束自己的生命的。
    全局/靜態存儲區,全局變量和靜態變量被分配到同一塊內存中,在以前的C語言中,全局變量又分為初始化的和未初始化的,在C++里面沒有這個區分了,他們共同占用同一塊內存區。
    常量存儲區,這是一塊比較特殊的存儲區,他們里面存放的是常量,不允許修改(當然,你要通過非正當手段也可以修改,而且方法很多)
明確區分堆與棧
    在bbs上,堆與棧的區分問題,似乎是一個永恒的話題,由此可見,初學者對此往往是混淆不清的,所以我決定拿他第一個開刀。
    首先,我們舉一個例子:
    void f() { int* p=new int[5]; }
    這條短短的一句話就包含了堆與棧,看到new,我們首先就應該想到,我們分配了一塊堆內存,那么指針p呢?他分配的是一塊棧內存,所以這句話的意思就是:在棧內存中存放了一個指向一塊堆內存的指針p。在程序會先確定在堆中分配內存的大小,然后調用operator new分配內存,然后返回這塊內存的首地址,放入棧中,他在VC6下的匯編代碼如下:
    00401028   push        14h
    0040102A   call        operator new (00401060)
    0040102F   add         esp,4
    00401032   mov         dword ptr [ebp-8],eax
    00401035   mov         eax,dword ptr [ebp-8]
    00401038   mov         dword ptr [ebp-4],eax
    這里,我們為了簡單并沒有釋放內存,那么該怎么去釋放呢?是delete p么?澳,錯了,應該是delete []p,這是為了告訴編譯器:我刪除的是一個數組,VC6就會根據相應的Cookie信息去進行釋放內存的工作。
    好了,我們回到我們的主題:堆和棧究竟有什么區別?
    主要的區別由以下幾點:
    1、管理方式不同;
    2、空間大小不同;
    3、能否產生碎片不同;
    4、生長方向不同;
    5、分配方式不同;
    6、分配效率不同;
    管理方式:對于棧來講,是由編譯器自動管理,無需我們手工控制;對于堆來說,釋放工作由程序員控制,容易產生memory leak。
    空間大小:一般來講在32位系統下,堆內存可以達到4G的空間,從這個角度來看堆內存幾乎是沒有什么限制的。但是對于棧來講,一般都是有一定的空間大小的,例如,在VC6下面,默認的棧空間大小是1M(好像是,記不清楚了)。當然,我們可以修改:   
    打開工程,依次操作菜單如下:Project->Setting->Link,在Category 中選中Output,然后在Reserve中設定堆棧的最大值和commit。
注意:reserve最小值為4Byte;commit是保留在虛擬內存的頁文件里面,它設置的較大會使棧開辟較大的值,可能增加內存的開銷和啟動時間。
    碎片問題:對于堆來講,頻繁的new/delete勢必會造成內存空間的不連續,從而造成大量的碎片,使程序效率降低。對于棧來講,則不會存在這個問題,因為棧是先進后出的隊列,他們是如此的一一對應,以至于永遠都不可能有一個內存塊從棧中間彈出,在他彈出之前,在他上面的后進的棧內容已經被彈出,詳細的可以參考數據結構,這里我們就不再一一討論了。
    生長方向:對于堆來講,生長方向是向上的,也就是向著內存地址增加的方向;對于棧來講,它的生長方向是向下的,是向著內存地址減小的方向增長。
    分配方式:堆都是動態分配的,沒有靜態分配的堆。棧有2種分配方式:靜態分配和動態分配。靜態分配是編譯器完成的,比如局部變量的分配。動態分配由alloca函數進行分配,但是棧的動態分配和堆是不同的,他的動態分配是由編譯器進行釋放,無需我們手工實現。
    分配效率:棧是機器系統提供的數據結構,計算機會在底層對棧提供支持:分配專門的寄存器存放棧的地址,壓棧出棧都有專門的指令執行,這就決定了棧的效率比較高。堆則是C/C++函數庫提供的,它的機制是很復雜的,例如為了分配一塊內存,庫函數會按照一定的算法(具體的算法可以參考數據結構/操作系統)在堆內存中搜索可用的足夠大小的空間,如果沒有足夠大小的空間(可能是由于內存碎片太多),就有可能調用系統功能去增加程序數據段的內存空間,這樣就有機會分到足夠大小的內存,然后進行返回。顯然,堆的效率比棧要低得多。
    從這里我們可以看到,堆和棧相比,由于大量new/delete的使用,容易造成大量的內存碎片;由于沒有專門的系統支持,效率很低;由于可能引發用戶態和核心態的切換,內存的申請,代價變得更加昂貴。所以棧在程序中是應用最廣泛的,就算是函數的調用也利用棧去完成,函數調用過程中的參數,返回地址,EBP和局部變量都采用棧的方式存放。所以,我們推薦大家盡量用棧,而不是用堆。
    雖然棧有如此眾多的好處,但是由于和堆相比不是那么靈活,有時候分配大量的內存空間,還是用堆好一些。
    無論是堆還是棧,都要防止越界現象的發生(除非你是故意使其越界),因為越界的結果要么是程序崩潰,要么是摧毀程序的堆、棧結構,產生以想不到的結果,就算是在你的程序運行過程中,沒有發生上面的問題,你還是要小心,說不定什么時候就崩掉,那時候debug可是相當困難的:)
    對了,還有一件事,如果有人把堆棧合起來說,那它的意思是棧,可不是堆,呵呵,清楚了?
static用來控制變量的存儲方式和可見性
       函數內部定義的變量,在程序執行到它的定義處時,編譯器為它在棧上分配空間,函數在棧上分配的空間在此函數執行結束時會釋放掉,這樣就產生了一個問題: 如果想將函數中此變量的值保存至下一次調用時,如何實現? 最容易想到的方法是定義一個全局的變量,但定義為一個全局變量有許多缺點,最明顯的缺點是破壞了此變量的訪問范圍(使得在此函數中定義的變量,不僅僅受此函數控制)。

       需要一個數據對象為整個類而非某個對象服務,同時又力求不破壞類的封裝性,即要求此成員隱藏在類的內部,對外不可見。

       static的內部機制:
       靜態數據成員要在程序一開始運行時就必須存在。因為函數在程序運行中被調用,所以靜態數據成員不能在任何函數內分配空間和初始化。
       這樣,它的空間分配有三個可能的地方,一是作為類的外部接口的頭文件,那里有類聲明;二是類定義的內部實現,那里有類的成員函數定義;三是應用程序的main()函數前的全局數據聲明和定義處。
      靜態數據成員要實際地分配空間,故不能在類的聲明中定義(只能聲明數據成員)。類聲明只聲明一個類的“尺寸和規格”,并不進行實際的內存分配,所以在類聲明中寫成定義是錯誤的。它也不能在頭文件中類聲明的外部定義,因為那會造成在多個使用該類的源文件中,對其重復定義。
      static被引入以告知編譯器,將變量存儲在程序的靜態存儲區而非棧上空間,靜態
數據成員按定義出現的先后順序依次初始化,注意靜態成員嵌套時,要保證所嵌套的成員已經初始化了。消除時的順序是初始化的反順序。

       static的優勢:
       可以節省內存,因為它是所有對象所公有的,因此,對多個對象來說,靜態數據成員只存儲一處,供所有對象共用。靜態數據成員的值對每個對象都是一樣,但它的值是可以更新的。只要對靜態數據成員的值更新一次,保證所有對象存取更新后的相同的值,這樣可以提高時間效率。

        引用靜態數據成員時,采用如下格式:
         <類名>::<靜態成員名>
    如果靜態數據成員的訪問權限允許的話(即public的成員),可在程序中,按上述格式
來引用靜態數據成員。

       PS:
      (1)類的靜態成員函數是屬于整個類而非類的對象,所以它沒有this指針,這就導致
了它僅能訪問類的靜態數據和靜態成員函數。
      (2)不能將靜態成員函數定義為虛函數。
      (3)由于靜態成員聲明于類中,操作于其外,所以對其取地址操作,就多少有些特殊
,變量地址是指向其數據類型的指針 ,函數地址類型是一個“nonmember函數指針”。

      (4)由于靜態成員函數沒有this指針,所以就差不多等同于nonmember函數,結果就
產生了一個意想不到的好處:成為一個callback函數,使得我們得以將C++和C-based X W
indow系統結合,同時也成功的應用于線程函數身上。
      (5)static并沒有增加程序的時空開銷,相反她還縮短了子類對父類靜態成員的訪問
時間,節省了子類的內存空間。
      (6)靜態數據成員在<定義或說明>時前面加關鍵字static。
      (7)靜態數據成員是靜態存儲的,所以必須對它進行初始化。
      (8)靜態成員初始化與一般數據成員初始化不同:
      初始化在類體外進行,而前面不加static,以免與一般靜態變量或對象相混淆;
      初始化時不加該成員的訪問權限控制符private,public等;
           初始化時使用作用域運算符來標明它所屬類;
           所以我們得出靜態數據成員初始化的格式:
         <數據類型><類名>::<靜態數據成員名>=<值>
      (9)為了防止父類的影響,可以在子類定義一個與父類相同的靜態變量,以屏蔽父類的影響。這里有一點需要注意:我們說靜態成員為父類和子類共享,但我們有重復定義了靜態成員,這會不會引起錯誤呢?不會,我們的編譯器采用了一種絕妙的手法:name-mangling 用以生成唯一的標志。

posted @ 2006-03-09 11:20 Beginning to 編程 閱讀(319) | 評論 (0)編輯 收藏

關于C++中構造函數的說明 /轉孫鑫VC++

在國內的C++圖書中,關于構造函數的說明,要么是錯誤的,要么沒有真正說清楚構造函數的作用,因此在我的視頻中,對構造函數的講解,也有一部分錯誤的敘述。對于C++構造函數一些錯誤認識的傳播,我也相當于起了推波助瀾的作用,在此反省一下,并給出正確的敘述。

(感謝西安軟件園的王先生為我指出錯誤,感謝網友backer幫助我找出正確的答案。)

在光盤VC02中,在介紹構造函數時,我說:“構造函數最重要的作用是創建對象本身,對象內存的分配由構造函數來完成的”,這句話是錯的,對象內存的分配和構造函數沒有關系,對象內存的分配是由編譯器來完成的,構造函數的作用是對對象本身做初始化工作,也就是給用戶提供初始化類中成員變量的一種方式,在類對象有虛表的情況下,構造函數還對虛表進行初始化。

另外,我說:“C++又規定,如果一個類沒有提供任何的構造函數,則C++提供一個默認的構造函數(由C++編譯器提供)”,這句話也是錯誤的,正確的是:

如果一個類中沒有定義任何的構造函數,那么編譯器只有在以下三種情況,才會提供默認的構造函數:
1、如果類有虛擬成員函數或者虛擬繼承父類(即有虛擬基類)時;
2、如果類的基類有構造函數(可以是用戶定義的構造函數,或編譯器提供的默認構造函數);
3、在類中的所有非靜態的對象數據成員,它們對應的類中有構造函數(可以是用戶定義的構造函數,或編譯器提供的默認構造函數)。

posted @ 2006-03-09 10:59 Beginning to 編程 閱讀(468) | 評論 (0)編輯 收藏

C++的多態性實現機制剖析 /轉孫鑫


 

――即VC++視頻第三課this指針詳細說明

作者:孫鑫 時間:2006年1月12日星期四

要更好地理解C++的多態性,我們需要弄清楚函數覆蓋的調用機制,因此,首先我們介紹一下函數的覆蓋。

1.   函數的覆蓋

我們先看一個例子:

1- 1

#include <iostream.h>

class animal

{

public:

       void sleep()

       {

              cout<<"animal sleep"<<endl;

       }

       void breathe()

       {

              cout<<"animal breathe"<<endl;

       }

};

class fish:public animal

{

public:

       void breathe()

       {

              cout<<"fish bubble"<<endl;

       }

};

void main()

{

       fish fh;

       animal *pAn=&fh;

       pAn->breathe();

}

       注意,在例1-1的程序中沒有定義虛函數。考慮一下例1-1的程序執行的結果是什么?

       答案是輸出:animal breathe

       在類fish中重寫了breathe()函數,我們可以稱為函數的覆蓋。在main()函數中首先定義了一個fish對象fh,接著定義了一個指向animal的指針變量pAn,將fh的地址賦給了指針變量pAn,然后利用該變量調用pAn->breathe()。許多學員往往將這種情況和C++的多態性搞混淆,認為fh實際上是fish類的對象,應該是調用fish類的breathe(),輸出“fish bubble”,然后結果卻不是這樣。下面我們從兩個方面來講述原因。

1、  編譯的角度

C++編譯器在編譯的時候,要確定每個對象調用的函數的地址,這稱為早期綁定(early binding),當我們將fish類的對象fh的地址賦給pAn時,C++編譯器進行了類型轉換,此時C++編譯器認為變量pAn保存就是animal對象的地址。當在main()函數中執行pAn->breathe()時,調用的當然就是animal對象的breathe函數。

2、  內存模型的角度

我們給出了fish對象內存模型,如下圖所示:







圖1- 1 fish類對象的內存模型

我們構造fish類的對象時,首先要調用animal類的構造函數去構造animal類的對象,然后才調用fish類的構造函數完成自身部分的構造,從而拼接出一個完整的fish對象。當我們將fish類的對象轉換為animal類型時,該對象就被認為是原對象整個內存模型的上半部分,也就是圖1-1中的“animal的對象所占內存”。那么當我們利用類型轉換后的對象指針去調用它的方法時,當然也就是調用它所在的內存中的方法。因此,出現圖2.13所示的結果,也就順理成章了。

2.   多態性和虛函數

正如很多學員所想,在例1-1的程序中,我們知道pAn實際指向的是fish類的對象,我們希望輸出的結果是魚的呼吸方法,即調用fish類的breathe方法。這個時候,就該輪到虛函數登場了。

前面輸出的結果是因為編譯器在編譯的時候,就已經確定了對象調用的函數的地址,要解決這個問題就要使用遲綁定(late binding)技術。當編譯器使用遲綁定時,就會在運行時再去確定對象的類型以及正確的調用函數。而要讓編譯器采用遲綁定,就要在基類中聲明函數時使用virtual關鍵字(注意,這是必須的,很多學員就是因為沒有使用虛函數而寫出很多錯誤的例子),這樣的函數我們稱為虛函數。一旦某個函數在基類中聲明為virtual,那么在所有的派生類中該函數都是virtual,而不需要再顯示的聲明為virtual

下面修改例1-1的代碼,將animal類中的breathe()函數聲明為virtual,如下:

1- 2

#include <iostream.h>

class animal

{

public:

       void sleep()

       {

              cout<<"animal sleep"<<endl;

       }

       virtual void breathe()

       {

              cout<<"animal breathe"<<endl;

       }

};

class fish:public animal

{

public:

       void breathe()

       {

              cout<<"fish bubble"<<endl;

       }

};

void main()

{

       fish fh;

       animal *pAn=&fh;

       pAn->breathe();

}

大家可以再次運行這個程序,你會發現結果是“fish bubble”,也就是根據對象的類型調用了正確的函數。

那么當我們將breathe()聲明為virtual時,在背后發生了什么呢?

編譯器在編譯的時候,發現animal類中有虛函數,此時編譯器會為每個包含虛函數的類創建一個虛表(即vtable),該表是一個一維數組,在這個數組中存放每個虛函數的地址。對于例1-2的程序,animal和fish類都包含了一個虛函數breathe(),因此編譯器會為這兩個類都建立一個虛表,如下圖所示:

圖1- 2 animal類和fish類的虛表

       那么如何定位虛表呢?編譯器另外還為每個類提供了一個虛表指針(即vptr),這個指針指向了對象的虛表。在程序運行時,根據對象的類型去初始化vptr,從而讓vptr正確的指向所屬類的虛表,從而在調用虛函數時,就能夠找到正確的函數。對于例1-2的程序,由于pAn實際指向的對象類型是fish,因此vptr指向的fish類的vtable,當調用pAn->breathe()時,根據虛表中的函數地址找到的就是fish類的breathe()函數。

正是由于每個對象調用的虛函數都是通過虛表指針來索引的,也就決定了虛表指針的正確初始化是非常重要的。換句話說,在虛表指針沒有正確初始化之前,我們不能夠去調用虛函數。那么虛表指針在什么時候,或者說在什么地方初始化呢?

答案是在構造函數中進行虛表的創建和虛表指針的初始化。還記得構造函數的調用順序嗎,在構造子類對象時,要先調用父類的構造函數,此時編譯器只“看到了”父類,并不知道后面是否后還有繼承者,它初始化父類的虛表指針,該虛表指針指向父類的虛表。當執行子類的構造函數時,子類的虛表指針被初始化,指向自身的虛表。對于例2-2的程序來說,當fish類的fh對象構造完畢后,其內部的虛表指針也就被初始化為指向fish類的虛表。在類型轉換后,調用pAn->breathe(),由于pAn實際指向的是fish類的對象,該對象內部的虛表指針指向的是fish類的虛表,因此最終調用的是fish類的breathe()函數。

要注意:對于虛函數調用來說,每一個對象內部都有一個虛表指針,該虛表指針被初始化為本類的虛表。所以在程序中,不管你的對象類型如何轉換,但該對象內部的虛表指針是固定的,所以呢,才能實現動態的對象函數調用,這就是C++多態性實現的原理。

總結(基類有虛函數):

1、  每一個類都有虛表。

2、  虛表可以繼承,如果子類沒有重寫虛函數,那么子類虛表中仍然會有該函數的地址,只不過這個地址指向的是基類的虛函數實現。如果基類3個虛函數,那么基類的虛表中就有三項(虛函數地址),派生類也會有虛表,至少有三項,如果重寫了相應的虛函數,那么虛表中的地址就會改變,指向自身的虛函數實現。如果派生類有自己的虛函數,那么虛表中就會添加該項。

3、  派生類的虛表中虛函數地址的排列順序和基類的虛表中虛函數地址排列順序相同。

3.   VC視頻第三課this指針說明

我在論壇的VC教學視頻版面發了帖子,是模擬MFC類庫的例子寫的,主要是說明在基類的構造函數中保存的this指針是指向子類的,我們在看一下這個例子:

例1- 3

#include <iostream.h>

class base;

base * pbase;

class base

{

public:

       base()

       {

              pbase=this;

              

       }

       virtual void fn()

       {

              cout<<"base"<<endl;

       }

};

class derived:public base

{

       void fn()

       {

              cout<<"derived"<<endl;

       }

};

derived aa;

void main()

{

       pbase->fn();

}

我在base類的構造函數中將this指針保存到pbase全局變量中。在定義全局對象aa,即調用derived aa;時,要調用基類的構造函數,先構造基類的部分,然后是子類的部分,由這兩部分拼接出完整的對象aa。這個this指針指向的當然也就是aa對象,那么我們main()函數中利用pbase調用fn(),因為pbase實際指向的是aa對象,而aa對象內部的虛表指針指向的是自身的虛表,最終調用的當然是derived類中的fn()函數。

在這個例子中,由于我的疏忽,在derived類中聲明fn()函數時,忘了加public關鍵字,導致聲明為了private(默認為private),但通過前面我們所講述的虛函數調用機制,我們也就明白了這個地方并不影響它輸出正確的結果。不知道這算不算C++的一個Bug,因為虛函數的調用是在運行時確定調用哪一個函數,所以編譯器在編譯時,并不知道pbase指向的是aa對象,所以導致這個奇怪現象的發生。如果你直接用aa對象去調用,由于對象類型是確定的(注意aa是對象變量,不是指針變量),編譯器往往會采用早期綁定,在編譯時確定調用的函數,于是就會發現fn()是私有的,不能直接調用。:)

許多學員在寫這個例子時,直接在基類的構造函數中調用虛函數,前面已經說了,在調用基類的構造函數時,編譯器只“看到了”父類,并不知道后面是否后還有繼承者,它只是初始化父類的虛表指針,讓該虛表指針指向父類的虛表,所以你看到結果當然不正確。只有在子類的構造函數調用完畢后,整個虛表才構建完畢,此時才能真正應用C++的多態性。換句話說,我們不要在構造函數中去調用虛函數,當然如果你只是想調用本類的函數,也無所謂。

4.   參考資料:

1、文章《在VC6.0中虛函數的實現方法》,作者:backer ,網址:

http://www.mybole.com.cn/bbs/dispbbs.asp?boardid=4&id=1012&star=1

2、書《C++編程思想》 機械工業出版社

5.   后記

本想再寫詳細些,發現時間不夠,總是有很多事情,在加上水平也有限,想想還是以后再說吧。不過我相信,這些內容也能夠幫助大家很好的理解了。也歡迎網友能夠繼續補充,大家可以鼓動鼓動backer,讓他從匯編的角度再給一個說明,哈哈,別說我說的。


posted @ 2006-03-09 10:56 Beginning to 編程 閱讀(378) | 評論 (0)編輯 收藏

我的職業

我畢業于東華大學服裝設計與工程,畢業后在一家服裝CAD公司做售后服務工程師.
離開這家公司后感覺,終于感覺又長大拉,也又老拉.

之后來到一家外資公司,還不錯.也沒有什么可忙的.
想到自己一直想編個小軟件(服裝繪圖方面)的,前一段時間
比較忙,現在可以靜心下來學習一下基本理論.就建立了這個Blog
也學學時髦吧.

今年1月我女兒生拉,真挺高興的.

posted @ 2006-03-04 19:17 Beginning to 編程 閱讀(288) | 評論 (0)編輯 收藏

僅列出標題
共3頁: 1 2 3 

導航

統計

常用鏈接

留言簿(4)

隨筆分類

隨筆檔案

文章檔案

相冊

BlogDev

搜索

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲女人av| 免费在线亚洲| 尤物yw午夜国产精品视频| 欧美视频精品在线观看| 欧美日韩免费在线观看| 欧美久久久久中文字幕| 欧美日韩免费高清| 国产精品黄视频| 国外成人性视频| 亚洲卡通欧美制服中文| 亚洲特黄一级片| 久久久久国产一区二区三区| 久久人人爽国产| 亚洲国产精品久久| 99精品国产在热久久| 午夜视频在线观看一区二区| 久久久久久久久久码影片| 免费一区二区三区| 国产精品久久国产精麻豆99网站| 国产亚洲一区在线| 日韩午夜三级在线| 久久精品国产精品亚洲精品| 欧美成人精品在线播放| 亚洲免费黄色| 久久精品视频免费观看| 欧美日韩国产一区二区三区地区| 国产深夜精品| 亚洲第一综合天堂另类专| 欧美视频免费| 美女网站久久| 久久综合狠狠综合久久综青草| 欧美大片免费观看在线观看网站推荐| 欧美精品二区| 国精产品99永久一区一区| 亚洲性人人天天夜夜摸| 蜜桃精品久久久久久久免费影院| 夜夜夜久久久| 欧美黄色大片网站| 亚洲三级视频在线观看| 久久夜色精品国产亚洲aⅴ| 一区二区av在线| 欧美性色aⅴ视频一区日韩精品| 狠狠久久综合婷婷不卡| 久久国产成人| 久久久久.com| 日韩午夜三级在线| 亚洲午夜在线观看| 国产在线播放一区二区三区| 久久一区国产| 欧美激情亚洲精品| 欧美在线观看一二区| 久久乐国产精品| 亚洲高清不卡av| 亚洲乱码国产乱码精品精可以看 | 亚洲精品一区二区三区不| 欧美激情国产日韩精品一区18| 99视频超级精品| 正在播放欧美一区| 伊人成人在线视频| 国产一区二区三区精品久久久 | 国内精品久久久久久久果冻传媒| 久久久久久网| 欧美日韩一区二| 久久男人av资源网站| 欧美—级a级欧美特级ar全黄| 亚洲欧美中文日韩v在线观看| 久久精品男女| 久久深夜福利| 国语精品一区| 欧美专区亚洲专区| 久久成年人视频| 国产精品萝li| 亚洲精品小视频在线观看| 91久久国产精品91久久性色| 欧美在线日韩在线| 久久久av水蜜桃| 亚洲精品免费观看| 久久久久久久久久看片| 免费91麻豆精品国产自产在线观看| 久久久夜夜夜| 亚洲全黄一级网站| 亚洲精品国产精品国自产在线 | 欧美激情国产精品| 在线观看国产日韩| 欧美电影在线播放| 一区二区三区www| 久久精品国产清高在天天线 | 欧美成人精品不卡视频在线观看| 久久人体大胆视频| 久久久噜噜噜久久| 老司机一区二区三区| 亚洲高清资源| 欧美色视频在线| 久久蜜桃资源一区二区老牛| 亚洲第一黄色| 免费精品99久久国产综合精品| 亚洲精品一区在线观看| 国产乱码精品1区2区3区| 看欧美日韩国产| 亚洲综合日韩中文字幕v在线| 免费视频久久| 午夜在线视频一区二区区别| 亚洲韩国精品一区| 国产一区二区三区自拍| 国产精品成人在线观看| 欧美日韩国产成人在线| 欧美黄色小视频| 免费亚洲一区二区| 久久久久国产精品一区二区| 在线亚洲国产精品网站| 99综合在线| 亚洲一区二区三区在线播放| 欧美激情第4页| 久久国产精品久久久| 亚洲欧美日本另类| 一本色道久久88综合亚洲精品ⅰ| 国产精品日韩精品欧美精品| 欧美成人综合一区| 久久久免费av| 欧美日韩国产综合新一区| 欧美日韩成人一区| 国产精品久久久99| 国产精品日韩欧美| 国内外成人免费激情在线视频 | 久久久噜噜噜| 麻豆视频一区二区| 国产精品国产馆在线真实露脸| 国产精品高清在线| 亚洲一区二区3| 欧美在线视屏| 国产精品久久久久久久久久久久久久 | 亚洲第一二三四五区| 亚洲精品久久久久| 亚洲综合色激情五月| 久久婷婷国产综合国色天香| 欧美精品入口| 狠狠色丁香婷婷综合| 在线视频精品| 欧美大片免费观看在线观看网站推荐| 亚洲人成毛片在线播放| 午夜视频一区在线观看| 久久福利资源站| 国产精品久久| 亚洲综合色网站| 一区电影在线观看| 欧美aⅴ99久久黑人专区| 国产日本亚洲高清| 日韩视频一区二区| 欧美1区2区| 久久亚洲综合网| 亚洲人线精品午夜| 欧美国产日韩免费| 欧美不卡福利| 日韩一级片网址| 夜夜嗨av一区二区三区网页 | 亚洲福利视频一区| 久久综合成人精品亚洲另类欧美| 国产一区白浆| 香蕉久久a毛片| 亚洲美女在线视频| 模特精品在线| 一区二区日韩免费看| 亚洲电影免费观看高清完整版| 久久激情一区| 久久三级福利| 亚洲第一免费播放区| 欧美第一黄色网| 欧美日韩福利视频| 亚洲一区二区三区久久| 午夜电影亚洲| 日韩一级精品视频在线观看| 亚洲天堂久久| 亚洲精品国产精品国自产在线 | 免费欧美视频| 国产精品黄色| 99ri日韩精品视频| 国内外成人免费视频| 亚洲电影免费在线| 国产精品亚洲激情| 久久婷婷一区| 欧美日本不卡| 免费久久99精品国产| 国语自产精品视频在线看一大j8| 亚洲欧洲精品一区二区三区波多野1战4| 国产精品视频网站| 小黄鸭视频精品导航| 欧美日韩一级大片网址| 性18欧美另类| 欧美精品播放| 欧美在线播放视频| 欧美大尺度在线| 午夜在线精品偷拍| 美日韩精品免费观看视频| 亚洲激情视频在线| 欧美岛国激情| 欧美日韩中文字幕精品| 亚洲激情网站| 欧美在线资源| 亚洲免费一级电影| 亚洲婷婷综合色高清在线|