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

山寨:不是最好的,是最適合我們的!歡迎體驗山寨 中文版MSDN

Blog @ Blog

當華美的葉片落盡,生命的脈絡才歷歷可見。 -- 聶魯達

常用鏈接

統計

積分與排名

BBS

Blog

Web

最新評論

C++中的虛函數(virtual function)

1.簡介
    虛函數是C++中用于實現多態(tài)(polymorphism)的機制。核心理念就是通過基類訪問派生類定義的函數。假設我們有下面的類層次:

class A
{
public:
    
virtual void foo() { cout << "A::foo() is called" << endl;}
}
;

class B: public A
{
public:
    
virtual void foo() { cout << "B::foo() is called" << endl;}
}
;


那么,在使用的時候,我們可以:

A * a = new B();
a->foo();       // 在這里,a雖然是指向A的指針,但是被調用的函數(foo)卻是B的!

    這個例子是虛函數的一個典型應用,通過這個例子,也許你就對虛函數有了一些概念。它虛就虛在所謂“推遲聯編”或者“動態(tài)聯編”上,一個類函數的調用并不是在編譯時刻被確定的,而是在運行時刻被確定的。由于編寫
代碼的時候并不能確定被調用的是基類的函數還是哪個派生類的函數,所以被成為“虛”函數。

    虛函數只能借助于指針或者引用來達到多態(tài)的效果,如果是下面這樣的代碼,則雖然是虛函數,但它不是多態(tài)的:

class A
{
public:
    
virtual void foo();
}
;

class B: public A
{
    
virtual void foo();
}
;

void bar()
{
    A a;
    a.foo();   
// A::foo()被調用
}


1.1 多態(tài)
    在
了解了虛函數的意思之后,再考慮什么是多態(tài)就很容易了。仍然針對上面的類層次,但是使用的方法變的復雜了一些:

void bar(A * a)
{
    a->foo();  // 被調用的是A::foo() 還是B::foo()?
}

因為foo()是個虛函數,所以在bar這個函數中,只根據這段代碼,無從確定這里被調用的是A::foo()還是B::foo(),但是可以肯定的說:如果a指向的是A類的實例,則A::foo()被調用,如果a指向的是B類的實例,則B::foo()被調用。

這種同一代碼可以產生不同效果的特點,被稱為“多態(tài)”。

1.2 多態(tài)有什么用?
    多態(tài)這么
神奇,但是能用來做什么呢?這個命題我難以用一兩句話概括,一般的C++教程(或者其它面向對象語言的教程)都用一個畫圖的例子來展示多態(tài)的用途,我就不再重復這個例子了,如果你不知道這個例子,隨便找本書應該都有介紹。我試圖從一個抽象的角度描述一下,回頭再結合那個畫圖的例子,也許你就更容易理解

    在面向對象的編程中,首先會針對數據進行抽象(確定基類)和繼承(確定派生類),構成類層次。這個類層次的使用者在使用它們的時候,如果仍然在需要基類的時候寫針對基類的代碼,在需要派生類的時候寫針對派生類的代碼,就等于類層次完全暴露在使用者面前。如果這個類層次有任何的改變(增加了新類),都需要使用者“知道”(針對新類寫代碼)。這樣就增加了類層次與其使用者之間的耦合,有人把這種情況列為程序中的“bad smell”之一。

    多態(tài)可以使程序員脫離這種窘境。再回頭看看1.1中的例子,bar()作為A-B這個類層次的使用者,它并不知道這個類層次中有多少個類,每個類都叫什么,但是一樣可以很好的工作,當有一個C類從A類派生出來后,bar()也不需要“知道”(修改)。這完全歸功于多態(tài)--編譯器針對虛函數產生了可以在運行時刻確定被調用函數的代碼。

1.3 如何“動態(tài)聯編”
    編譯器是如何針對虛函數產生可以再運行時刻確定被調用函數的代碼呢?也就是說,虛函數實際上是如何被編譯器處理的呢?Lippman在深度探索C++對象模型[1]中的不同章節(jié)講到了幾種方式,這里把“標準的”方式簡單介紹一下。

    我所說的“標準”方式,也就是所謂的“VTABLE”機制。編譯器發(fā)現一個類中有被聲明為virtual的函數,就會為其搞一個虛函數表,也就是VTABLE。VTABLE實際上是一個函數指針的數組,每個虛函數占用這個數組的一個slot。一個類只有一個VTABLE,不管它有多少個實例。派生類有自己的VTABLE,但是派生類的VTABLE與基類的VTABLE有相同的函數排列順序,同名的虛函數被放在兩個數組的相同位置上。在創(chuàng)建類實例的時候,編譯器還會在每個實例的內存布局中增加一個vptr字段,該字段指向本類的VTABLE。通過這些手段,編譯器在看到一個虛函數調用的時候,就會將這個調用改寫,針對1.1中的例子:

class A
{
public:
    
virtual void foo();
}
;

class B: public A
{
    
virtual void foo();
}
;

void bar()
{
    A a;
    a.foo();   
// A::foo()被調用
}



    因為派生類和基類的foo()函數具有相同的VTABLE索引,而他們的vptr又指向不同的VTABLE,因此通過這樣的方法可以在運行時刻決定調用哪個foo()函數。

    雖然實際情況遠非這么簡單,但是基本原理大致如此。

1.4 overload和override
    虛函數總是在派生類中被改寫,這種改寫被稱為“override”。我經?;煜?#8220;overload”和“override”這兩個單詞。但是隨著各類C++的書越來越多,后來的程序員也許不會再犯我犯過的錯誤了。但是我打算澄清一下:

override是指派生類重寫基類的虛函數,就象我們前面B類中重寫了A類中的foo()函數。重寫的函數必須有一致的參數表和返回值(C++標準允許返回值不同的情況,這個我會在“語法”部分簡單介紹,但是很少編譯器支持這個feature)。這個單詞好象一直沒有什么合適的中文詞匯來對應,有人譯為“覆蓋”,還貼切一些。
overload約定成俗的被翻譯為“重載”。是指編寫一個與已有函數同名但是參數表不同的函數。例如一個函數即可以接受整型數作為參數,也可以接受浮點數作為參數。
2. 虛函數的語法
    虛函數的標志是“virtual”關鍵字。

2.1 使用virtual關鍵字
    考慮下面的類層次:

class A
{
public:
    
virtual void foo();
}
;

class B: public A
{
public:
    
void foo();    // 沒有virtual關鍵字!
}
;

class C: public B  // 從B繼承,不是從A繼承!
{
public:
    
void foo();    // 也沒有virtual關鍵字!
}
;


    這種情況下,B::foo()是虛函數,C::foo()也同樣是虛函數。因此,可以說,基類聲明的虛函數,在派生類中也是虛函數,即使不再使用virtual關鍵字。

2.2 純虛函數
    如下聲明表示一個函數為純虛函數:

class A
{
public:
    
virtual void foo()=0;   // =0標志一個虛函數為純虛函數
}
;


    一個函數聲明為純虛后,純虛函數的意思是:我是一個抽象類!不要把我實例化!純虛函數用來規(guī)范派生類的行為,實際上就是所謂的“接口”。它告訴使用者,我的派生類都會有這個函數。

2.3 虛析構函數
    析構函數也可以是虛的,甚至是純虛的。例如:

class A
{
public:
    
virtual ~A()=0;   // 純虛析構函數
}
;


    當一個類打算被用作其它類的基類時,它的析構函數必須是虛的??紤]下面的例子:

class A
{
public:
    A() 
{ ptra_ = new char[10];}
    
~A() { delete[] ptra_;}        // 非虛析構函數
private:
    
char * ptra_;
}
;

class B: public A
{
public:
    B() 
{ ptrb_ = new char[20];}
    
~B() { delete[] ptrb_;}
private:
    
char * ptrb_;
}
;

void foo()
{
    A 
* a = new B;
    delete a;
}


    在這個例子中,程序也許不會象你想象的那樣運行,在執(zhí)行delete a的時候,實際上只有A::~A()被調用了,而B類的析構函數并沒有被調用!這是否有點兒可怕?

    如果將上面A::~A()改為virtual,就可以保證B::~B()也在delete a的時候被調用了。因此基類的析構函數都必須是virtual的。

    純虛的析構函數并沒有什么作用,是虛的就夠了。通常只有在希望將一個類變成抽象類(不能實例化的類),而這個類又沒有合適的函數可以被純虛化的時候,可以使用純虛的析構函數來達到目的。

2.4 虛構造函數?
    構造函數不能是虛的。

3. 虛函數使用技巧 3.1 private的虛函數
    考慮下面的例子:

class A
{
public:
    
void foo() { bar();}
private:
    
virtual void bar() }
}
;

class B: public A
{
private:
    
virtual void bar() }
}
;


    在這個例子中,雖然bar()在A類中是private的,但是仍然可以出現在派生類中,并仍然可以與public或者protected的虛函數一樣產生多態(tài)的效果。并不會因為它是private的,就發(fā)生A::foo()不能訪問B::bar()的情況,也不會發(fā)生B::bar()對A::bar()的override不起作用的情況。

    這種寫法的語意是:A告訴B,你最好override我的bar()函數,但是你不要管它如何使用,也不要自己調用這個函數。

3.2 構造函數和析構函數中的虛函數調用
    一個類的虛函數在它自己的構造函數和析構函數中被調用的時候,它們就變成普通函數了,不“虛”了。也就是說不能在構造函數和析構函數中讓自己“多態(tài)”。例如:

class A
{
public:
    A() 
{ foo();}        // 在這里,無論如何都是A::foo()被調用!
    ~A() { foo();}       // 同上
    virtual void foo();
}
;

class B: public A
{
public:
    
virtual void foo();
}
;

void bar()
{
    A 
* a = new B;
    delete a;
}


    如果你希望delete a的時候,會導致B::foo()被調用,那么你就錯了。同樣,在new B的時候,A的構造函數被調用,但是在A的構造函數中,被調用的是A::foo()而不是B::foo()。

3.3 多繼承中的虛函數 3.4 什么時候使用虛函數
    在你設計一個基類的時候,如果發(fā)現一個函數需要在派生類里有不同的表現,那么它就應該是虛的。從設計的角度講,出現在基類中的虛函數是接口,出現在派生類中的虛函數是接口的具體實現。通過這樣的方法,就可以將對象的行為抽象化。

    以設計模式[2]中Factory Method模式為例,Creator的factoryMethod()就是虛函數,派生類override這個函數后,產生不同的Product類,被產生的Product類被基類的AnOperation()函數使用?;惖腁nOperation()函數針對Product類進行操作,當然Product類一定也有多態(tài)(虛函數)。

    另外一個例子就是集合操作,假設你有一個以A類為基類的類層次,又用了一個std::vector<A *>來保存這個類層次中不同類的實例指針,那么你一定希望在對這個集合中的類進行操作的時候,不要把每個指針再cast回到它原來的類型(派生類),而是希望對他們進行同樣的操作。那么就應該將這個“一樣的操作”聲明為virtual。

    現實中,遠不只我舉的這兩個例子,但是大的原則都是我前面說到的“如果發(fā)現一個函數需要在派生類里有不同的表現,那么它就應該是虛的”。這句話也可以反過來說:“如果你發(fā)現基類提供了虛函數,那么你最好override它”。

4.參考資料
[1] 深度探索C++對象模型,Stanley B.Lippman,侯捷譯

[2] Design Patterns, Elements of Reusable Object-Oriented Software, GOF

posted on 2008-04-27 01:15 isabc 閱讀(484) 評論(0)  編輯 收藏 引用 所屬分類: C++基礎

廣告信息(免費廣告聯系)

中文版MSDN:
歡迎體驗

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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黑人| 亚洲美女免费精品视频在线观看| 免费欧美在线视频| 久久不射2019中文字幕| 国产一区二区高清| 久久精品国产视频| 欧美一区成人| 激情综合久久| 欧美大片一区二区| 欧美激情国产高清| 亚洲一区二区三区激情| 亚洲一区二区高清| 国产在线视频欧美| 久久综合一区| 欧美va亚洲va香蕉在线| 亚洲人午夜精品| 亚洲乱码国产乱码精品精可以看 | 午夜激情一区| 欧美一区二区成人| 伊人夜夜躁av伊人久久| 免费亚洲电影| 欧美激情综合五月色丁香小说| 一本色道久久综合亚洲精品高清| 亚洲精选国产| 国产伦精品一区二区三区视频黑人| 久久成人亚洲| 免费欧美电影| 亚洲女人av| 久久av在线| 亚洲三级电影在线观看 | 欧美视频福利| 久久精品一区蜜桃臀影院| 久久久99免费视频| 亚洲最新在线视频| 性感少妇一区| 亚洲国产精品电影| 亚洲综合另类| 91久久精品国产| 99天天综合性| 在线播放中文一区| 夜夜夜久久久| 亚洲国产cao| 亚洲色诱最新| 亚洲国产高清在线| 亚洲一区欧美激情| 亚洲欧洲日产国码二区| 亚洲制服av| 亚洲六月丁香色婷婷综合久久| 亚洲女同精品视频| 亚洲精品资源| 久久精品理论片| 亚洲私人影院在线观看| 久久久久久久综合日本| 欧美91大片| 国产综合久久久久影院| 欧美韩国一区| 国产亚洲精品资源在线26u| 91久久精品视频| 国内精品久久久久久影视8| 亚洲美女电影在线| 亚洲国产精品va在线看黑人动漫| 一区二区av| 亚洲日本电影| 卡通动漫国产精品| 欧美在线视频观看| 欧美日韩1080p| 亚洲国产日韩欧美在线99| 国产日韩在线视频| 亚洲一区二区三区777| 亚洲最新视频在线| 欧美国产亚洲视频| 亚洲高清av在线| 亚洲成在人线av| 久久精品理论片| 久久久999| 国产精品女人网站| 亚洲先锋成人| 亚洲欧美日韩另类精品一区二区三区 | 久久嫩草精品久久久精品| 国产精品久久久999| 99成人精品| 99精品欧美一区二区三区| 欧美精品亚洲二区| 亚洲精品社区| 一区二区高清视频在线观看| 欧美日产一区二区三区在线观看| 亚洲欧洲视频| 一区二区高清视频在线观看| 欧美日韩精品免费观看视频| 亚洲精品日韩在线| 亚洲一区二区三区在线| 欧美日韩在线播放三区| 在线亚洲一区观看| 欧美在线免费观看亚洲| 激情五月综合色婷婷一区二区| 欧美一区午夜精品| 欧美成人第一页| 一本大道av伊人久久综合| 欧美视频一区二区三区…| 一本色道久久综合狠狠躁篇的优点| 亚洲欧美成人在线| 国内精品久久久久影院优| 麻豆成人小视频| 亚洲精品一区中文| 欧美在线免费看| 亚洲高清资源| 欧美性大战久久久久久久| 新狼窝色av性久久久久久| 米奇777超碰欧美日韩亚洲| 日韩视频一区二区三区在线播放| 欧美日韩精品综合| 亚洲在线视频观看| 老巨人导航500精品| 99在线热播精品免费| 国产欧美日韩中文字幕在线| 久久久www成人免费精品| 亚洲高清免费视频| 午夜精品电影| 亚洲国产综合在线| 国产精品国产a| 美国成人直播| 欧美日韩成人网| 欧美亚洲视频在线观看| 欧美电影资源| 午夜精品久久久久久久99黑人| 激情另类综合| 国产精品久久久久久久久搜平片| 欧美在线亚洲在线| 亚洲精品乱码久久久久久日本蜜臀 | 在线视频你懂得一区二区三区| 国产精品视频大全| 欧美国产日韩a欧美在线观看| 亚洲在线视频观看| 亚洲精品久久久蜜桃| 久久青草福利网站| 亚洲女人天堂成人av在线| 亚洲激情影院| 好看的日韩视频| 国产精品亚洲片夜色在线| 欧美区二区三区| 久久视频一区| 亚洲专区免费| 亚洲免费av观看| 亚洲国产日韩欧美在线99| 久久视频一区二区| 欧美在线高清视频| 亚洲男人第一网站| 亚洲午夜激情| 9色porny自拍视频一区二区| 韩日在线一区| 国产三区二区一区久久| 国产精品扒开腿做爽爽爽软件| 欧美激情视频免费观看| 久久久久综合| 欧美在线免费看| 小黄鸭视频精品导航| 午夜激情一区| 亚洲午夜精品久久久久久app| 亚洲人成啪啪网站| 亚洲国产另类精品专区| 免费一级欧美片在线播放| 久久综合中文| 蜜桃精品一区二区三区| 久久只有精品| 蜜桃av一区二区三区| 麻豆精品在线观看| 久久影院亚洲| 蜜桃久久精品乱码一区二区| 欧美91大片| 欧美高清自拍一区| 亚洲大片在线观看| 免费观看成人| 亚洲电影免费| 91久久在线视频| 亚洲免费观看高清完整版在线观看| 91久久精品网| 日韩亚洲欧美高清| 亚洲摸下面视频| 欧美一区二区精品久久911| 欧美一区二区三区在| 久久精品日产第一区二区三区 | 亚洲国产美女精品久久久久∴| 伊人成人开心激情综合网| 亚洲国产第一页| 亚洲欧洲视频| 亚洲一区视频在线观看视频| 欧美有码视频| 蜜臀av性久久久久蜜臀aⅴ四虎| 欧美成年人网| 一区二区电影免费观看|