關(guān)于構(gòu)造函數(shù)的調(diào)用順序
作者:王姍姍, 華清遠(yuǎn)見(jiàn)嵌入式學(xué)院講師。
在說(shuō)構(gòu)造函數(shù)之前我們得先弄明白幾個(gè)問(wèn)題,首先是什么是類的構(gòu)造函數(shù),什么是類的成員對(duì)象,什么是基類,然后我們?cè)賮?lái)說(shuō)構(gòu)造函數(shù)的調(diào)用順序。
1、 類的構(gòu)造函數(shù)
構(gòu)造函數(shù)的功能主要用于在類的對(duì)象創(chuàng)建時(shí)定義初始化的狀態(tài)。它沒(méi)有返回值,也不能用void來(lái)修飾,這就保證了它不僅什么也不用自動(dòng)返回,而且根本不能有任何選擇。構(gòu)造函數(shù)不能被直接調(diào)用,必須通過(guò)new運(yùn)算符在創(chuàng)建對(duì)象時(shí)才會(huì)自動(dòng)調(diào)用,一般方法在程序執(zhí)行到它的時(shí)候被調(diào)用。一個(gè)類可以有多個(gè)構(gòu)造函數(shù),根據(jù)其參數(shù)個(gè)數(shù)的不同或參數(shù)類型的不同來(lái)區(qū)分它們,即構(gòu)造函數(shù)的重載。
2、 類的成員對(duì)象
類包含數(shù)據(jù)成員和成員函數(shù)。當(dāng)類中的成員數(shù)據(jù)是另一個(gè)類的對(duì)象時(shí),我們就說(shuō)這個(gè)類是該類的成員對(duì)象。
3、 基類
通過(guò)繼承機(jī)制,可以利用已有的數(shù)據(jù)類型來(lái)定義新的數(shù)據(jù)類型。所定義的新的數(shù)據(jù)類型不僅擁有新定義的成員,而且還同時(shí)擁有舊的成員。我們稱已存在的用來(lái)派生新類的類為基類,又稱為父類。
4、 構(gòu)造函數(shù)的調(diào)用順序
我們來(lái)看下面一段代碼:
class B1
{
public:
B1(int i) {cout<<"constructing B1 "<<i<<endl;}
};
class B2
{
public:
B2(int j) {cout<<"constructing B2 "<<j<<endl;}
};
class B3
{
public:
B3( ){cout<<"constructing B3 *"<<endl;}
};
class C: public B2, public B1, public B3
{
public:
C(int a, int b, int c, int d):B1(a),memberB2(d),memberB1(c),B2(b){}
private:
B1 memberB1;
B2 memberB2;
B3 memberB3;
};
void main( )
{ C obj(1,2,3,4); }
運(yùn)行后的結(jié)果如下:
constructing B2 2
constructing B1 1
constructing B3 *
constructing B1 3
constructing B2 4
constructing B3 *
為什么會(huì)有以上的結(jié)果?
眾所周知構(gòu)造函數(shù)的執(zhí)行次序如下:
調(diào)用基類構(gòu)造函數(shù),調(diào)用順序按照他們的繼承時(shí)聲明的順序。
調(diào)用內(nèi)嵌成員對(duì)象的構(gòu)造函數(shù),調(diào)用順序按照他們?cè)陬愔新暶鞯捻樞颉?/FONT>
派生類的構(gòu)造函數(shù)體中的內(nèi)容。
析構(gòu)函數(shù)的調(diào)用順序相反。
那么再來(lái)看以上的例子就很容易理解了。B2、B1、B3是C的基類,按照上述的順序,我們先要構(gòu)造基類,然后才是子對(duì)象,最后是其本身的構(gòu)造函數(shù)所以先要執(zhí)行這三個(gè)類的構(gòu)造函數(shù)。在構(gòu)造時(shí)按照他們?cè)陬愔械捻樞?,首先調(diào)用B2的構(gòu)造函數(shù)
B2(int j) {cout<<"constructing B2 "<<j<<endl;}
由于在默認(rèn)參數(shù)列表
C(int a, int b, int c, int d):B1(a),memberB2(d),memberB1(c),B2(b){}
中,將b的值傳給了B2的構(gòu)造函數(shù),b為2,故打印出:
constructing B2 2
接下來(lái)要構(gòu)造的是B1了。顯然在C的默認(rèn)參數(shù)構(gòu)造列表中將a的值傳給了B1,
所以打印出:
constructing B1 1
B3在構(gòu)造時(shí)沒(méi)有傳遞參數(shù),調(diào)用B3( ){cout<<"constructing B3 *"<<endl;}
打印出:
cout<<"constructing B3 *
這時(shí)基類的構(gòu)造函數(shù)已經(jīng)執(zhí)行完畢,接著該處理內(nèi)嵌成員對(duì)象的構(gòu)造函數(shù)了。
我們看到C類有三個(gè)對(duì)象:B1 memberB1;B2 memberB2;B3 memberB3;,按照構(gòu)造函數(shù)的調(diào)用順序,我們需要按照他們?cè)陬愔新暶鞯捻樞騺?lái)分別構(gòu)造memberB1、memberB2、 memberB3。在默認(rèn)的參數(shù)列表中,用c來(lái)構(gòu)造了memberB1,用d來(lái)構(gòu)造memberB2,
故打印出:
constructing B1 3
constructing B2 4
constructing B3 *
最后調(diào)用本身的構(gòu)造函數(shù),由于函數(shù)體為空,故什么也沒(méi)有打印出來(lái)。
總結(jié)下來(lái),我們必須明確的是當(dāng)一個(gè)類繼承與基類,并且自身還包含有其他類的成員對(duì)象的時(shí)候,構(gòu)造函數(shù)的調(diào)用順序?yàn)椋赫{(diào)用基類的構(gòu)造函數(shù)->調(diào)用成員對(duì)象的構(gòu)造函數(shù)->調(diào)用自身的構(gòu)造函數(shù)。構(gòu)造函數(shù)的調(diào)用次序完全不受構(gòu)造函數(shù)初始化列表的表達(dá)式中的次序影響,與基類的聲明次數(shù)和成員對(duì)象在函數(shù)中的聲明次序有關(guān)。
“本文由華清遠(yuǎn)見(jiàn)http://www.embedu.org/index.htm提供”
來(lái)源:華清遠(yuǎn)見(jiàn)1次