深入解析C++中派生類的構(gòu)造函數(shù)
基類的構(gòu)造函數(shù)不能被繼承,在聲明派生類時(shí),對(duì)繼承過(guò)來(lái)的成員變量的初始化工作也要由派生類的構(gòu)造函數(shù)來(lái)完成。所以在設(shè)計(jì)派生類的構(gòu)造函數(shù)時(shí),不僅要考慮派生類新增的成員變量,還要考慮基類的成員變量,要讓它們都被初始化。
解決這個(gè)問(wèn)題的思路是:在執(zhí)行派生類的構(gòu)造函數(shù)時(shí),調(diào)用基類的構(gòu)造函數(shù)。
下面的例子展示了如何在派生類的構(gòu)造函數(shù)中調(diào)用基類的構(gòu)造函數(shù)。
#include<iostream> using namespace std; //基類 class People{ protected: char *name; int age; public: People(char*, int); }; People::People(char *name, int age): name(name), age(age){} //派生類 class Student: public People{ private: float score; public: Student(char*, int, float); void display(); }; //調(diào)用了基類的構(gòu)造函數(shù) Student::Student(char *name, int age, float score): People(name, age){ this->score = score; } void Student::display(){ cout<<name<<"的年齡是"<<age<<",成績(jī)是"<<score<<endl; } int main(){ Student stu("小明", 16, 90.5); stu.display(); return 0; }
運(yùn)行結(jié)果為:
小明的年齡是16,成績(jī)是90.5
請(qǐng)注意代碼第23行:
Student::Student(char *name, int age, float score): People(name, age)
這是派生類 Student 的構(gòu)造函數(shù)的寫法。冒號(hào)前面是派生類構(gòu)造函數(shù)的頭部,這和我們以前介紹的構(gòu)造函數(shù)的形式一樣,但它的形參列表包括了初始化基類和派生類的成員變量所需的數(shù)據(jù);冒號(hào)后面是對(duì)基類構(gòu)造函數(shù)的調(diào)用,這和普通構(gòu)造函數(shù)的參數(shù)初始化表非常類似。
實(shí)際上,你可以將對(duì)基類構(gòu)造函數(shù)的調(diào)用和參數(shù)初始化表放在一起,如下所示:
Student::Student(char *name, int age, float score): People(name, age), score(score){}
基類構(gòu)造函數(shù)和初始化表用逗號(hào)隔開。
需要注意的是:冒號(hào)后面是對(duì)基類構(gòu)造函數(shù)的調(diào)用,而不是聲明,所以括號(hào)里的參數(shù)是實(shí)參,它們不但可以是派生類構(gòu)造函數(shù)總參數(shù)表中的參數(shù),還可以是局部變量、常量等。如下所示:
Student::Student(char *name, int age, float score): People("李磊", 20)
基類構(gòu)造函數(shù)調(diào)用規(guī)則
事實(shí)上,通過(guò)派生類創(chuàng)建對(duì)象時(shí)必須要調(diào)用基類的構(gòu)造函數(shù),這是語(yǔ)法規(guī)定。也就是說(shuō),定義派生類構(gòu)造函數(shù)時(shí)最好指明基類構(gòu)造函數(shù);如果不指明,就調(diào)用基類的默認(rèn)構(gòu)造函數(shù)(不帶參數(shù)的構(gòu)造函數(shù));如果沒有默認(rèn)構(gòu)造函數(shù),那么編譯失敗。
請(qǐng)看下面的例子:
#include<iostream> using namespace std; //基類 class People{ protected: char *name; int age; public: People(); People(char*, int); }; People::People(){ this->name = "xxx"; this->age = 0; } People::People(char *name, int age): name(name), age(age){} //派生類 class Student: public People{ private: float score; public: Student(); Student(char*, int, float); void display(); }; Student::Student(){ this->score = 0.0; } Student::Student(char *name, int age, float score): People(name, age){ this->score = score; } void Student::display(){ cout<<name<<"的年齡是"<<age<<",成績(jī)是"<<score<<endl; } int main(){ Student stu1; stu1.display(); Student stu2("小明", 16, 90.5); stu2.display(); return 0; }
運(yùn)行結(jié)果:
xxx的年齡是0,成績(jī)是0 小明的年齡是16,成績(jī)是90.5
創(chuàng)建對(duì)象 stu1 時(shí),執(zhí)行派生類的構(gòu)造函數(shù) Student::Student(),它并沒有指明要調(diào)用基類的哪一個(gè)構(gòu)造函數(shù),從運(yùn)行結(jié)果可以很明顯地看出來(lái),系統(tǒng)默認(rèn)調(diào)用了不帶參數(shù)的構(gòu)造函數(shù),也就是 People::People()。
創(chuàng)建對(duì)象 stu2 時(shí),執(zhí)行派生類的構(gòu)造函數(shù) Student::Student(char *name, int age, float score),它指明了基類的構(gòu)造函數(shù)。
在第31行代碼中,如果將 People(name, age) 去掉,也會(huì)調(diào)用默認(rèn)構(gòu)造函數(shù),stu2.display() 的輸出結(jié)果將變?yōu)椋?br /> xxx的年齡是0,成績(jī)是90.5
如果將基類 People 中不帶參數(shù)的構(gòu)造函數(shù)刪除,那么會(huì)發(fā)生編譯錯(cuò)誤,因?yàn)閯?chuàng)建對(duì)象 stu1 時(shí)沒有調(diào)用基類構(gòu)造函數(shù)。
總結(jié):如果基類有默認(rèn)構(gòu)造函數(shù),那么在派生類構(gòu)造函數(shù)中可以不指明,系統(tǒng)會(huì)默認(rèn)調(diào)用;如果沒有,那么必須要指明,否則系統(tǒng)不知道如何調(diào)用基類的構(gòu)造函數(shù)。
構(gòu)造函數(shù)的調(diào)用順序
為了搞清這個(gè)問(wèn)題,我們不妨先來(lái)看一個(gè)例子:
#include<iostream> using namespace std; //基類 class People{ protected: char *name; int age; public: People(); People(char*, int); }; People::People(): name("xxx"), age(0){ cout<<"PeoPle::People()"<<endl; } People::People(char *name, int age): name(name), age(age){ cout<<"PeoPle::People(char *, int)"<<endl; } //派生類 class Student: public People{ private: float score; public: Student(); Student(char*, int, float); }; Student::Student(): score(0.0){ cout<<"Student::Student()"<<endl; } Student::Student(char *name, int age, float score): People(name, age), score(score){ cout<<"Student::Student(char*, int, float)"<<endl; } int main(){ Student stu1; cout<<"--------------------"<<endl; Student stu2("小明", 16, 90.5); return 0; }
運(yùn)行結(jié)果:
PeoPle::People() Student::Student() -------------------- PeoPle::People(char *, int) Student::Student(char*, int, float)
從運(yùn)行結(jié)果可以清楚地看到,當(dāng)創(chuàng)建派生類對(duì)象時(shí),先調(diào)用基類構(gòu)造函數(shù),再調(diào)用派生類構(gòu)造函數(shù)。如果繼承關(guān)系有好幾層的話,例如:
A --> B --> C
那么則創(chuàng)建C類對(duì)象時(shí),構(gòu)造函數(shù)的執(zhí)行順序?yàn)椋?br />
A類構(gòu)造函數(shù) --> B類構(gòu)造函數(shù) --> C類構(gòu)造函數(shù)
構(gòu)造函數(shù)的調(diào)用順序是按照繼承的層次自頂向下、從基類再到派生類的。
C++有子對(duì)象的派生類的構(gòu)造函數(shù)
類的數(shù)據(jù)成員不但可以是標(biāo)準(zhǔn)型(如int、char)或系統(tǒng)提供的類型(如string),還可以包含類對(duì)象,如可以在聲明一個(gè)類時(shí)包含這樣的數(shù)據(jù)成員:
Student s1; //Student是已聲明的類名,s1是Student類的對(duì)象
這時(shí),s1就是類對(duì)象中的內(nèi)嵌對(duì)象,稱為子對(duì)象(subobject),即對(duì)象中的對(duì)象。
那么,在對(duì)數(shù)據(jù)成員初始化時(shí)怎樣對(duì)子對(duì)象初始化呢?請(qǐng)仔細(xì)分析下面程序,特別注意派生類構(gòu)造函數(shù)的寫法。
[例] 包含子對(duì)象的派生類的構(gòu)造函數(shù)。為了簡(jiǎn)化程序以易于閱讀,這里設(shè)基類Student的數(shù)據(jù)成員只有兩個(gè),即num和name。
#include <iostream> #include <string> using namespace std; class Student//聲明基類 { public: //公用部分 Student(int n, string nam ) //基類構(gòu)造函數(shù),與例11.5相同 { num=n; name=nam; } void display( ) //成員函數(shù),輸出基類數(shù)據(jù)成員 { cout<<"num:"<<num<<endl<<"name:"<<name<<endl; } protected: //保護(hù)部分 int num; string name; }; class Student1: public Student //聲明公用派生類Student1 { public: Student1(int n, string nam,int n1, string nam1,int a, string ad):Student(n,nam),monitor(n1,nam1) //派生類構(gòu)造函數(shù) { age=a; addr=ad; } void show( ) { cout<<"This student is:"<<endl; display(); //輸出num和name cout<<"age: "<<age<<endl; //輸出age cout<<"address: "<<addr<<endl<<endl; //輸出addr } void show_monitor( ) //成員函數(shù),輸出子對(duì)象 { cout<<endl<<"Class monitor is:"<<endl; monitor.display( ); //調(diào)用基類成員函數(shù) } private: //派生類的私有數(shù)據(jù) Student monitor; //定義子對(duì)象(班長(zhǎng)) int age; string addr; }; int main( ) { Student1 stud1(10010,"Wang-li",10001,"Li-sun",19,"115 Beijing Road,Shanghai"); stud1.show( ); //輸出學(xué)生的數(shù)據(jù) stud1.show_monitor(); //輸出子對(duì)象的數(shù)據(jù) return 0; }
運(yùn)行時(shí)的輸出如下:
This student is: num: 10010 name: Wang-li age: 19 address:115 Beijing Road,Shanghai Class monitor is: num:10001 name:Li-sun
請(qǐng)注意在派生類Student1中有一個(gè)數(shù)據(jù)成員:
Student monitor; //定義子對(duì)象 monitor(班長(zhǎng))
“班長(zhǎng)”的類型不是簡(jiǎn)單類型(如int、char、float等),它是Student類的對(duì)象。我們知道, 應(yīng)當(dāng)在建立對(duì)象時(shí)對(duì)它的數(shù)據(jù)成員初始化。那么怎樣對(duì)子對(duì)象初始化呢?顯然不能在聲明派生類時(shí)對(duì)它初始化(如Student monitor(10001, "Li-fun");),因?yàn)轭愂浅橄箢愋?,只是一個(gè)模型,是不能有具體的數(shù)據(jù)的,而且每一個(gè)派生類對(duì)象的子對(duì)象一般是不相同的(例如學(xué)生A、B、C的班長(zhǎng)是A,而學(xué)生D、E、F的班長(zhǎng)是F)。因此子對(duì)象的初始化是在建立派生類時(shí)通過(guò)調(diào)用派生類構(gòu)造函數(shù)來(lái)實(shí)現(xiàn)的。
派生類構(gòu)造函數(shù)的任務(wù)應(yīng)該包括3個(gè)部分:
- 對(duì)基類數(shù)據(jù)成員初始化;
- 對(duì)子對(duì)象數(shù)據(jù)成員初始化;
- 對(duì)派生類數(shù)據(jù)成員初始化。
程序中派生類構(gòu)造函數(shù)首部如下:
Student1(int n, string nam,int n1, string nam1,int a, string ad): Student(n,nam),monitor(n1,nam1)
在上面的構(gòu)造函數(shù)中有6個(gè)形參,前兩個(gè)作為基類構(gòu)造函數(shù)的參數(shù),第3、第4個(gè)作為子對(duì)象構(gòu)造函數(shù)的參數(shù),第5、第6個(gè)是用作派生類數(shù)據(jù)成員初始化的。
歸納起來(lái),定義派生類構(gòu)造函數(shù)的一般形式為: 派生類構(gòu)造函數(shù)名(總參數(shù)表列): 基類構(gòu)造函數(shù)名(參數(shù)表列), 子對(duì)象名(參數(shù)表列)
{
派生類中新增數(shù)成員據(jù)成員初始化語(yǔ)句
}
執(zhí)行派生類構(gòu)造函數(shù)的順序是:
調(diào)用基類構(gòu)造函數(shù),對(duì)基類數(shù)據(jù)成員初始化;
調(diào)用子對(duì)象構(gòu)造函數(shù),對(duì)子對(duì)象數(shù)據(jù)成員初始化;
再執(zhí)行派生類構(gòu)造函數(shù)本身,對(duì)派生類數(shù)據(jù)成員初始化。
派生類構(gòu)造函數(shù)的總參數(shù)表列中的參數(shù),應(yīng)當(dāng)包括基類構(gòu)造函數(shù)和子對(duì)象的參數(shù)表列中的參數(shù)?;悩?gòu)造函數(shù)和子對(duì)象的次序可以是任意的,如上面的派生類構(gòu)造函數(shù)首部可以寫成
Student1(int n, string nam,int n1, string nam1,int a, string ad): monitor(n1,nam1),Student(n,nam)
編譯系統(tǒng)是根據(jù)相同的參數(shù)名(而不是根據(jù)參數(shù)的順序)來(lái)確立它們的傳遞關(guān)系的。但是習(xí)慣上一般先寫基類構(gòu)造函數(shù)。
如果有多個(gè)子對(duì)象,派生類構(gòu)造函數(shù)的寫法依此類推,應(yīng)列出每一個(gè)子對(duì)象名及其參數(shù)表列。
相關(guān)文章
C++基礎(chǔ)知識(shí)之運(yùn)算符重載詳解
這篇文章主要為大家詳細(xì)介紹了C++基礎(chǔ)知識(shí)之運(yùn)算符重載,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助2022-02-02C語(yǔ)言大小端字節(jié)序存儲(chǔ)模式深入解讀
我們知道,當(dāng)編譯器執(zhí)行 “創(chuàng)建變量” 這一代碼時(shí),會(huì)在內(nèi)存中開辟空間相應(yīng)的空間來(lái)存儲(chǔ)變量值。而對(duì)于整型變量而言,變量值又是以二進(jìn)制補(bǔ)碼的形式存放的2022-09-09一文搞懂C++中的四種強(qiáng)制類型轉(zhuǎn)換
很多朋友向小編了解C語(yǔ)言中怎么進(jìn)行強(qiáng)制類型轉(zhuǎn)換呢?在這小編告訴大家強(qiáng)制類型轉(zhuǎn)換可以分為兩種,一種是隱式類型轉(zhuǎn)換一種是顯示類型轉(zhuǎn)換,下面通過(guò)示例代碼給大家介紹下,需要的朋友參考下吧2021-07-07C語(yǔ)言例題之輸出1000以內(nèi)的所有完數(shù)
完數(shù)是一些特殊的自然數(shù),它所有的真因子(即除了自身以外的約數(shù))的和(即因子函數(shù)),恰好等于它本身,如果一個(gè)數(shù)恰好等于它的因子之和,則稱該數(shù)為“完數(shù)”,這篇文章主要給大家介紹了關(guān)于C語(yǔ)言例題之輸出1000以內(nèi)的所有完數(shù)的相關(guān)資料,需要的朋友可以參考下2022-11-11實(shí)現(xiàn)一個(gè)random?shuffle算法示例
這篇文章主要為大家介紹了實(shí)現(xiàn)一個(gè)random?shuffle算法示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-05-05