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