C++初始化列表 多態(tài)

初始化列表

特點(diǎn)

  • 一種便捷的初始化成員變量的方式
  • 只能在構(gòu)造函數(shù)中
      Person(int age, int height): m_age(age), m_height(height) { }

初始化列表與默認(rèn)參數(shù)配合使用

     Person(int age = 0, int height = 0): m_age(age),   m_height(height) { }

如果函數(shù)的聲明和實(shí)現(xiàn)是分離的

  • 初始化列表只能寫(xiě)在函數(shù)的實(shí)現(xiàn)中
  • 默認(rèn)參數(shù)只能寫(xiě)在函數(shù)的聲明中

父類(lèi)的構(gòu)造函數(shù)

  • 子類(lèi)的構(gòu)造函數(shù)默認(rèn)會(huì)調(diào)用父類(lèi)的無(wú)參構(gòu)造函數(shù)
  • 如果子類(lèi)的構(gòu)造函數(shù)顯示地調(diào)用了父類(lèi)的有慘構(gòu)造函數(shù)呆细,就不會(huì)再去默認(rèn)調(diào)用父類(lèi)的無(wú)參構(gòu)造函數(shù)
  • 如果父類(lèi)缺少無(wú)參構(gòu)造函數(shù)坟比,子類(lèi)的構(gòu)造函數(shù)必須顯式調(diào)用父類(lèi)的有參構(gòu)造函數(shù)

繼承體系下的構(gòu)造函數(shù)示例

struct Person {
    int m_age;
    
    Person(): Person(0) {}
    
    Person(int age): m_age(age) { }
};


struct Student: Person {
    int m_no;
    Student(): Student(0, 0) {
        
    }
    
    Student(int age, int no): Person(age), m_no(no) {
        
    }
};

父類(lèi)指針果元、子類(lèi)指針

  • 父類(lèi)指針可以指向子類(lèi)對(duì)象,是安全的(繼承方式必須是public)
  • 子類(lèi)指針指父類(lèi)對(duì)象是不安全的
class Person {
public:
    int m_age;
};

class Student : public Person {
public:
    int m_score;
};

void test() {
    Person *person = new Student();
    // 可以訪問(wèn)m_score甚脉, m_age; 安全
    person->m_age = 10;
    
    Student *stu = (Student*)new Person();
    // 訪問(wèn)不到 m_score, 不安全
    stu->m_score = 10;
}

多態(tài)

  • 默認(rèn)情況下,編譯器只會(huì)根據(jù)指針類(lèi)型調(diào)用對(duì)應(yīng)的函數(shù)婶希,不存在多態(tài)

特性

  • 同一個(gè)操作作用于不同的對(duì)象耕腾,可以有不同的解釋见剩,產(chǎn)生不同的執(zhí)行結(jié)果
  • 在運(yùn)行時(shí),可以識(shí)別出真正的對(duì)象類(lèi)型扫俺,調(diào)用對(duì)應(yīng)子類(lèi)中的函數(shù)

多態(tài)的要素

  • 有繼承關(guān)系
  • 子類(lèi)重寫(xiě)父類(lèi)的成員函數(shù)(override)
  • 父類(lèi)指針指向子類(lèi)對(duì)象
  • 利用父類(lèi)調(diào)用重寫(xiě)的成員函數(shù)

多態(tài)分為兩類(lèi)

  • 靜態(tài)多態(tài):函數(shù)重載和運(yùn)算符重載屬于靜態(tài)多態(tài)苍苞,復(fù)用函數(shù)名
  • 動(dòng)態(tài)多態(tài):派生類(lèi)和虛函數(shù)實(shí)現(xiàn)運(yùn)行時(shí)多態(tài)

靜態(tài)多態(tài)個(gè)動(dòng)態(tài)多態(tài)的區(qū)別

  • 靜態(tài)多態(tài)的函數(shù)地址早綁定-編譯階段確定函數(shù)地址
  • 動(dòng)態(tài)多態(tài)的函數(shù)地址晚綁定-運(yùn)行階段確定函數(shù)地址
class Animal {
public:
    virtual void run() {
        cout << "Animal::run()" << endl;
    }
};

class Dog: public Animal {
public:
    void run() override {
        cout << "Dog::run()" << endl;
    }
};

class ErHa: public Dog {
public:
    void run() override {
        cout << "ErHa::run()" << endl;
    }
};

void test() {
    Dog *dog = new Dog();
    dog->run();
    
    Animal * animal0 = new Dog();
    animal0->run();
    
    Animal * animal1 = new ErHa();
    animal1->run();
}

虛函數(shù)

  • C++中的多態(tài)通過(guò)虛函數(shù)(virtual function)來(lái)實(shí)現(xiàn)
  • 虛函數(shù):被virtual修飾的成員函數(shù)
  • 只要在父類(lèi)中聲明為虛函數(shù),子類(lèi)中重寫(xiě)的函數(shù)也自動(dòng)變成虛函數(shù)(也就是說(shuō)子類(lèi)可以省略virtual關(guān)鍵字)

虛表

  • 虛函數(shù)的實(shí)現(xiàn)原理是虛表牵舵,這個(gè)虛表里面存儲(chǔ)著最終需要調(diào)用的函數(shù)地址柒啤,這個(gè)虛表也叫虛函數(shù)表
class Animal {
public:
   virtual void run() {
       cout << "Animal::run()" << endl;
   }
     virtual void speak() {
       cout <<"Animal::speak()" << endl;
   }
};

class Cat: public Animal {
public:
    int m_life;
    Cat(): m_life(0) {}
    
    void run() override {
        cout << "Cat::run()" << endl;
    }
void speak() override {
        // 先執(zhí)行父類(lèi)的成員函數(shù)
        Animal::speak();倦挂、
        // 再執(zhí)行自己的一些操作
        cout << "Cat::Speak()" << endl;
    }
};

虛表的x86環(huán)境圖

image.png
  • 由上圖可知,所有的Cat對(duì)象只對(duì)應(yīng)一份虛表担巩。
  • 具有虛函數(shù)的類(lèi)方援,在對(duì)象創(chuàng)建時(shí),會(huì)在其內(nèi)存布局的時(shí)候涛癌,拿出4個(gè)字節(jié)的內(nèi)存犯戏,用來(lái)存儲(chǔ)該對(duì)象對(duì)應(yīng)的函數(shù)虛表地址(vfptr: 虛函數(shù)指針),x86環(huán)境中拳话,一個(gè)虛函數(shù)地址占用4個(gè)字節(jié)先匪。

虛表的匯編分析

   // 調(diào)用Cat::run
   // 取出cat指針變量里面存儲(chǔ)的地址值
   // eax里面存放的是cat對(duì)象的地址
   mov eax, dword ptr [cat]
   // 取出Cat對(duì)象的前面4個(gè)字節(jié)給edx
   // edx里面存儲(chǔ)的是虛表的地址
   mov edx, dword ptr [eax]
   // 取出虛表中的前面4個(gè)字節(jié)給eax
   // eax存放的就是Cat::run的函數(shù)地址
   mov eax, dword ptr [edx]
   call eax
   
   // 調(diào)用Cat::speak
   // 取出cat指針變量里面存儲(chǔ)的地址值
   // eax里面存放的是cat對(duì)象的地址
   mov eax, dword ptr [cat]
   // 取出Cat對(duì)象的前面4個(gè)字節(jié)給edx
   // edx里面存儲(chǔ)的是虛表的地址
   mov edx, dword ptr [eax]
   // 取出虛表中的后面4個(gè)字節(jié)給eax
   // eax存放的就是Cat::speak的函數(shù)地址
   mov eax, dword ptr [edx]
   call eax

虛析構(gòu)函數(shù)

  • 含有虛函數(shù)的類(lèi),應(yīng)該將析構(gòu)函數(shù)聲明為虛函數(shù)(虛析構(gòu)函數(shù))
  • delete 父類(lèi)指針時(shí)弃衍,才會(huì)調(diào)用子類(lèi)的析構(gòu)函數(shù)呀非,保證析構(gòu)的完整性
  • 虛析構(gòu)或純純虛析構(gòu)就是用來(lái)解決父類(lèi)指針釋放子類(lèi)對(duì)象
  • 如果子類(lèi)中沒(méi)有堆區(qū)數(shù)據(jù),可以不寫(xiě)為虛析構(gòu)或純虛析構(gòu)
  • 擁有純虛析構(gòu)函數(shù)的類(lèi)屬于抽象類(lèi)
class Animal {
public:
    virtual void run() {
        cout << "Animal::run()" << endl;
    }
    virtual void speak() {
        cout <<"Animal::speak()" << endl;
    }
    
   // 虛析構(gòu)
    virtual ~Animal() {
        
    }
  // 純虛析構(gòu)
  virtual ~Animal() = 0;

};


class Cat: public Animal {
public:
    int m_life;
    Cat(): m_life(0) {}
    
    void run() override {
        cout << "Cat::run()" << endl;
    }
    
    void speak() override {
        // 先執(zhí)行父類(lèi)的成員函數(shù)
        Animal::speak();
        // 再執(zhí)行自己的一些操作
        cout << "Cat::Speak()" << endl;
    }
    
    ~Cat() {
        cout << "Cat::~Cat()" << endl;
    }
};

純虛函數(shù)

  • 定義:沒(méi)有函數(shù)體且初始化為0的虛函數(shù)镜盯,用來(lái)定義接口規(guī)范
  • 抽象類(lèi)
    • 含有虛函數(shù)的類(lèi)岸裙,不可以實(shí)例化
    • 抽象類(lèi)也可以包含非純虛函數(shù)
    • 如果父類(lèi)是抽象類(lèi),子類(lèi)沒(méi)有完全實(shí)現(xiàn)純虛函數(shù)速缆,那么這個(gè)子類(lèi)依然是抽象類(lèi)
class Animal {
public:
    virtual void run() = 0;
    
    virtual void speak() = 0;
    
    virtual ~Animal() { }
};
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末降允,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子艺糜,更是在濱河造成了極大的恐慌剧董,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,042評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件破停,死亡現(xiàn)場(chǎng)離奇詭異翅楼,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)辱挥,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,996評(píng)論 2 384
  • 文/潘曉璐 我一進(jìn)店門(mén)犁嗅,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人晤碘,你說(shuō)我怎么就攤上這事褂微。” “怎么了园爷?”我有些...
    開(kāi)封第一講書(shū)人閱讀 156,674評(píng)論 0 345
  • 文/不壞的土叔 我叫張陵宠蚂,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我童社,道長(zhǎng)求厕,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,340評(píng)論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮呀癣,結(jié)果婚禮上美浦,老公的妹妹穿的比我還像新娘。我一直安慰自己项栏,他們只是感情好浦辨,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,404評(píng)論 5 384
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著沼沈,像睡著了一般流酬。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上列另,一...
    開(kāi)封第一講書(shū)人閱讀 49,749評(píng)論 1 289
  • 那天芽腾,我揣著相機(jī)與錄音,去河邊找鬼页衙。 笑死摊滔,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的店乐。 我是一名探鬼主播惭载,決...
    沈念sama閱讀 38,902評(píng)論 3 405
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼响巢!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起棒妨,我...
    開(kāi)封第一講書(shū)人閱讀 37,662評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤踪古,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后券腔,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體伏穆,經(jīng)...
    沈念sama閱讀 44,110評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,451評(píng)論 2 325
  • 正文 我和宋清朗相戀三年纷纫,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了枕扫。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,577評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡辱魁,死狀恐怖烟瞧,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情染簇,我是刑警寧澤参滴,帶...
    沈念sama閱讀 34,258評(píng)論 4 328
  • 正文 年R本政府宣布,位于F島的核電站锻弓,受9級(jí)特大地震影響砾赔,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,848評(píng)論 3 312
  • 文/蒙蒙 一暴心、第九天 我趴在偏房一處隱蔽的房頂上張望妓盲。 院中可真熱鬧,春花似錦专普、人聲如沸悯衬。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,726評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)甚亭。三九已至,卻和暖如春击胜,著一層夾襖步出監(jiān)牢的瞬間亏狰,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,952評(píng)論 1 264
  • 我被黑心中介騙來(lái)泰國(guó)打工偶摔, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留暇唾,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,271評(píng)論 2 360
  • 正文 我出身青樓辰斋,卻偏偏與公主長(zhǎng)得像策州,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子宫仗,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,452評(píng)論 2 348