C++中多繼承建構函式呼叫順序
class B1 {public: B1(int i) {cout<<"consB1"<<i<<endl;} };//定義基類B1 class B2 {public: B2(int j) {cout<<"consB2"<<j<<endl;} };//定義基類B2 class B3 { public: B3() {cout<<"consB3 *"<<endl;} };//定義基類B3 class C: public B2, public B1, public B3 {public: C(int a,int b,int c,int d,int e) :B1(a),memberB2(d),memberB1(c),B2(b) {m=e; cout<<"consC"<<endl;} private: B1 memberB1; B2 memberB2; B3 memberB3; int m; };//繼承類C void main() { C obj(1,2,3,4,5); }//主函式
執行結果:consB2 2 consB1 1 consB3 * consB1 3 consB2 4 consB3 * consC //先按照繼承順序:B2,B1,B3
//第一步:先繼承B2,在初始化列表裡找到B2(b),列印"constB22"
//第二步:再繼承B1,在初始化列表裡找到B1(a),列印"constB11"
//第三步:又繼承B3,在初始化列表裡找不到B3(x), 則呼叫B3裡的預設建構函式B3(),列印"constB3 *"
//再按照資料成員定義順序:memberB1, memberB2, memberB3
//第四步:在初始化列表裡找到memberB1(c),初始化一個B1物件,用c為引數,則呼叫B1的建構函式,列印"constB13"
//第五步:在初始化列表裡找到memberB2(d),初始化一個B2物件,用d為引數,則呼叫B2的建構函式,列印"constB24"
//第六步:在初始化列表裡找不到memberB3(x),則呼叫B3裡的預設建構函式B3(),列印"constB3 *"
//最後完成本物件初始化的剩下部分,也就是C自己的建構函式的函式體:{m=e; cout<<"consC"<<endl;}
//第七步:列印"consC"
回到你的主要問題:為什麼會有兩次B3*出現?
第一次是由於繼承了B3,雖然在C的建構函式的初始化列表裡你沒看到B3(x)或者B3(),但並不代表B3的建構函式沒有在發揮作用。事實上,B3被隱性初始化了,因為B3的建構函式沒有引數,所以寫不寫B3()都無所謂,這裡恰好省略了。B1,B2則都是顯性初始化,因為它們都需要引數。第二次是因為C有資料成員memberB3,又一次,你沒有在C的建構函式的初始化列表裡看到你希望出現的memberB3(),很顯然,這又是一次隱性初始化。B3的建構函式再次被暗中呼叫。每一次B3的建構函式被呼叫,都會打印出“consB3 *”。兩次被呼叫,自然列印兩次“consB3 *”。