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

大龍的博客

常用鏈接

統(tǒng)計

最新評論

注意--------------

class parent
{
public:
    parent() : m_p(this)
    {
        m_p->prints();            //輸出parent!(特殊的地方)    多態(tài)不能在構(gòu)造中起作用---具體說明如下
    }
    ~parent(){}

    virtual void prints()
    {
        std::cout << "parent!" << std::endl;
    }
 
    virtual void cout()
    {
        m_p->prints();           //多態(tài),輸出child!
    }
    parent *m_p;
};

class child : public parent
{
public:
    child()
    {
        this->prints();         //輸出child!
    }
    ~child(){}

    virtual void prints()
    {
        std::cout << "child!" << std::endl;
    }
};


int _tmain(int argc, _TCHAR* argv[])
{
    child onechild;
    onechild.cout();
}

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

來自編程愛好者

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

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的指針,但是被調(diào)用的函數(shù)(foo)卻是B的!

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

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

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

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

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

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

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

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

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

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

    在面向?qū)ο蟮木幊讨校紫葧槍?shù)據(jù)進行抽象(確定基類)和繼承(確定派生類),構(gòu)成類層次。這個類層次的使用者在使用它們的時候,如果仍然在需要基類的時候?qū)戓槍惖拇a,在需要派生類的時候?qū)戓槍ε缮惖拇a,就等于類層次完全暴露在使用者面前。如果這個類層次有任何的改變(增加了新類),都需要使用者“知道”(針對新類寫代碼)。這樣就增加了類層次與其使用者之間的耦合,有人把這種情況列為程序中的“bad smell”之一。

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

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

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

void bar(A * a)
{
    a->foo();
}

會被改寫為:

void bar(A * a)
{
    (a->vptr[1])();
}

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

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

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

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

2.1 使用virtual關(guān)鍵字
    考慮下面的類層次:

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

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

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

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

2.2 純虛函數(shù)
    如下聲明表示一個函數(shù)為純虛函數(shù):

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

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

2.3 虛析構(gòu)函數(shù)
    析構(gòu)函數(shù)也可以是虛的,甚至是純虛的。例如:

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

    當一個類打算被用作其它類的基類時,它的析構(gòu)函數(shù)必須是虛的。考慮下面的例子:

class A
{
public:
    A() { ptra_ = new char[10];}
    ~A() { delete[] ptra_;}        // 非虛析構(gòu)函數(shù)
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()被調(diào)用了,而B類的析構(gòu)函數(shù)并沒有被調(diào)用!這是否有點兒可怕?

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

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

2.4 虛構(gòu)造函數(shù)?
    構(gòu)造函數(shù)不能是虛的。

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

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

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

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

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

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

class A
{
public:
    A() { foo();}        // 在這里,無論如何都是A::foo()被調(diào)用!
    ~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()被調(diào)用,那么你就錯了。同樣,在new B的時候,A的構(gòu)造函數(shù)被調(diào)用,但是在A的構(gòu)造函數(shù)中,被調(diào)用的是A::foo()而不是B::foo()。

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

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

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

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

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

posted on 2007-05-13 00:36 大龍 閱讀(127) 評論(2)  編輯 收藏 引用

評論

# re: 注意-------------- 2007-07-05 16:03 luckbill

子類要實例化,首先調(diào)用的是父類的構(gòu)造函數(shù),此時的this指針是指向父類的函數(shù)表的,也就會調(diào)用父類的print函數(shù)了。  回復  更多評論   

# re: 注意-------------- 2007-07-09 18:17 大龍1

嗯。  回復  更多評論   


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


青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美区在线观看| 国产一级精品aaaaa看| 国产精品三级视频| 亚洲国产精品电影| 久久久久久一区二区| 欧美伊人影院| 亚洲网站在线| 亚洲欧美在线免费| 欧美亚州一区二区三区 | 久久久久一区二区三区| 亚洲精品中文字| 欧美电影打屁股sp| 性久久久久久久久久久久| 亚洲清纯自拍| 国产精品久久久久aaaa九色| 欧美国产先锋| 国产精品高清在线观看| 亚洲国产欧美一区| 免费国产自线拍一欧美视频| 亚洲夜晚福利在线观看| 亚洲国产精品传媒在线观看| 久久精品国产99国产精品澳门| 国产精品视频免费观看www| 亚洲免费一在线| 久久躁日日躁aaaaxxxx| 久久久99免费视频| 亚洲少妇最新在线视频| 欧美黑人国产人伦爽爽爽| 麻豆9191精品国产| 欧美电影免费观看| 亚洲午夜精品一区二区三区他趣| 亚洲在线网站| 美女视频黄免费的久久| 国产日本欧美一区二区| 久久婷婷影院| 亚洲第一福利在线观看| 一区二区三区精品在线| 久久久噜噜噜久噜久久| 亚洲黄色av一区| 国产精品一区二区久久久| 亚洲欧美日韩另类精品一区二区三区| 亚洲裸体在线观看| 亚洲综合国产激情另类一区| 欧美日韩免费区域视频在线观看| 一区二区三区高清视频在线观看| 欧美刺激午夜性久久久久久久| 国内精品亚洲| 欧美在线观看一二区| 一本高清dvd不卡在线观看| 一区二区亚洲| 国产精品盗摄一区二区三区| 国产欧美日本一区二区三区| 亚洲午夜精品网| 欧美在线看片a免费观看| 国产亚洲激情在线| 久久精品1区| 最近中文字幕日韩精品| 亚洲自拍啪啪| 日韩一级精品| 久久久久久久综合狠狠综合| 亚洲经典视频在线观看| 国产在线麻豆精品观看| 亚洲免费高清| 亚洲高清123| 亚洲精品一区久久久久久| 欧美一区二区性| 欧美深夜影院| 国产欧美日韩三区| 蜜桃伊人久久| 欧美视频官网| 99国产精品久久久久久久成人热| 一二三四社区欧美黄| 狠狠干综合网| 国产亚洲一区二区三区| 一区二区国产日产| 国产精品夜夜夜一区二区三区尤| 亚洲国产成人av| 国产综合婷婷| 国产一区二区三区精品久久久| 国产精品美女久久久久久2018| 亚洲一级免费视频| 国产欧美一区二区白浆黑人| 久久国产精彩视频| 欧美在线观看网站| 久久经典综合| 亚洲素人一区二区| 一本久道久久综合婷婷鲸鱼| 欧美日韩影院| 欧美一级视频一区二区| 亚洲成色www8888| 一区二区三区日韩精品| 香蕉成人伊视频在线观看 | 久久一区中文字幕| 狠狠色丁香婷婷综合久久片| 国内精品久久久久久久影视麻豆| 亚洲精品自在久久| 午夜亚洲性色福利视频| 亚洲国产欧美国产综合一区| 久久精品91久久久久久再现| 欧美一区二区三区四区在线| 在线一区二区三区四区五区| 18成人免费观看视频| 99成人精品| 亚洲第一网站免费视频| 久久亚洲风情| 欧美亚洲三区| 欧美在线视频网站| 亚洲第一偷拍| 亚洲视频播放| 亚洲色在线视频| 亚洲欧美另类在线| 欧美在线高清| 久久亚洲综合| 精久久久久久久久久久| 精品粉嫩aⅴ一区二区三区四区| 蜜桃av综合| 欧美日韩国产一中文字不卡| 国产精品天美传媒入口| 国产一区二区三区久久久| 一区二区三区日韩欧美精品| 中文日韩在线视频| 久久久国产视频91| 蜜桃视频一区| 新狼窝色av性久久久久久| 免费久久99精品国产自| 国产精品美女在线| 亚洲与欧洲av电影| 久久综合网络一区二区| 亚洲欧美综合v| 亚洲精品一区二区三区av| 欧美成人视屏| 一区二区在线观看视频在线观看| 欧美一级日韩一级| 久久久99国产精品免费| 一区二区三区亚洲| 亚洲欧美清纯在线制服| 欧美高清视频www夜色资源网| 午夜精品偷拍| 欧美伦理a级免费电影| 亚洲欧美精品伊人久久| 99re66热这里只有精品4| 欧美在线|欧美| 午夜久久黄色| 蘑菇福利视频一区播放| 亚洲国产综合在线看不卡| 麻豆成人91精品二区三区| 欧美久久久久| 亚洲图片欧洲图片av| 久久综合给合| 欧美v日韩v国产v| 亚洲一区在线播放| 久久久精品网| 亚洲视频999| 欧美深夜影院| 久久这里有精品15一区二区三区| 亚洲激情专区| 国产一级一区二区| 99re8这里有精品热视频免费| 欧美三区视频| 午夜精彩国产免费不卡不顿大片| 欧美精品国产一区二区| 亚洲尤物在线视频观看| 久久久久久久一区二区| 亚洲网站啪啪| 欧美四级伦理在线| 99综合精品| av成人免费在线观看| 欧美激情中文字幕一区二区| 欧美一区二区精品久久911| 久久国产主播精品| 久久高清福利视频| 国产日韩视频| 亚洲欧美韩国| 久久久久久综合| 国产日本欧美一区二区三区在线| 久久成人免费电影| 国产精品你懂得| 久久激情中文| 亚洲人成77777在线观看网| 狠狠综合久久av一区二区老牛| 卡一卡二国产精品| 久久亚洲一区二区| 国产一区二区精品久久99| 欧美一级久久久| 久久欧美中文字幕| 日韩一级精品视频在线观看| 国产婷婷色一区二区三区在线| 欧美在线免费看| 一区二区欧美在线观看| 欧美福利精品| 亚洲永久网站| 亚洲精品美女免费| 久久国产精品99精品国产| 国产精品久久久久久久久| 久久婷婷久久一区二区三区| 99精品欧美一区二区蜜桃免费| 亚洲黄色在线视频| 国产精品高清在线| 久久精品亚洲国产奇米99| 麻豆乱码国产一区二区三区|