何謂初始化列表
與其他函數不同,構造函數除了有名字,參數列表和函數體之外,還可以有初始化列表,初始化列表以冒號開頭,后跟一系列以逗號分隔的初始化字段。在C++中,struct和class的唯一區別是默認的克訪問性不同,而這里我們不考慮訪問性的問題,所以下面的代碼都以struct來演示。
初始化階段
所有類類型(class type)的成員都會在初始化階段初始化,即使該成員沒有出現在構造函數的初始化列表中。
計算階段
一般用于執行構造函數體內的賦值操作,下面的代碼定義兩個結構體,其中Test1有構造函數,拷貝構造函數及賦值運算符,為的是方便查看結果。Test2是個測試類,它以Test1的對象為成員,我們看一下Test2的構造函數是怎么樣執行的。
Test1(const Test1& t1) // 拷貝構造函數
{
cout << "Copy constructor for Test1" << endl ;
this->a = t1.a ;
}
Test1& operator = (const Test1& t1) // 賦值運算符
{
cout << "assignment for Test1" << endl ;
this->a = t1.a ;
return *this;
}
int a ;
};
struct Test2
{
Test1 test1 ;
Test2(Test1 &t1)
{
test1 = t1 ;
}
};
解釋一下,第一行輸出對應調用代碼中第一行,構造一個Test1對象。第二行輸出對應Test2構造函數中的代碼,用默認的構造函數初始化對象test1,這就是所謂的初始化階段。第三行輸出對應Test1的賦值運算符,對test1執行賦值操作,這就是所謂的計算階段。
為什么使用初始化列表
初始化類的成員有兩種方式,一是使用初始化列表,二是在構造函數體內進行賦值操作。使用初始化列表主要是基于性能問題,對于內置類型,如int, float等,使用初始化類表和在構造函數體內初始化差別不是很大,但是對于類類型來說,最好使用初始化列表,為什么呢?由上面的測試可知,使用初始化列表少了一次調用默認構造函數的過程,這對于數據密集型的類來說,是非常高效的。同樣看上面的例子,我們使用初始化列表來實現Test2的構造函數
第一行輸出對應 調用代碼的第一行。第二行輸出對應Test2的初始化列表,直接調用拷貝構造函數初始化test1,省去了調用默認構造函數的過程。所以一個好的原則是,能使用初始化列表的時候盡量使用初始化列表。
哪些東西必須放在初始化列表中
除了性能問題之外,有些時場合初始化列表是不可或缺的,以下幾種情況時必須使用初始化列表
常量成員,因為常量只能初始化不能賦值,所以必須放在初始化列表里面
引用類型,引用必須在定義的時候初始化,并且不能重新賦值,所以也要寫在初始化列表里面
沒有默認構造函數的類類型,因為使用初始化列表可以不必調用默認構造函數來初始化,而是直接調用拷貝構造函數初始化。
對于沒有默認構造函數的類,我們看一個例子。
struct Test2
{
Test1 test1 ;
Test2(Test1 &t1)
{
test1 = t1 ;
}
};
1. 調用Test1的默認構造函數來初始化test1
2. 調用Test1的賦值運算符給test1賦值
但是由于Test1沒有默認的構造函數,所謂第一步無法執行,故而編譯錯誤。正確的代碼如下,使用初始化列表代替賦值操作。
新聞熱點
疑難解答
圖片精選