我們定義一個如下的Person類:
class Person {
public:
Person() { } //default constructor function
Person(string name, string phone, string addr)
{
m_name = name; //想采用賦值初始化數(shù)據(jù)成員
m_phone = phone;
m_addr = addr;
}
private:
const string m_name;
const string m_phone;
const string m_addr;
};
編譯后發(fā)現(xiàn)這個類的第二個帶參數(shù)的構造函數(shù)是錯誤的。我們創(chuàng)建一個Person對象:
Person p("marcky", "13233232", "cqupt"); //調(diào)用帶參數(shù)的構造函數(shù)創(chuàng)建一個Person對象
創(chuàng)建對象的過程分為了兩步:
一、從內(nèi)存中分配實際的空間給對象p,其三個字符串對象的數(shù)據(jù)成員是調(diào)用的默認構造函數(shù)初始化為空。也就說,此時為止,對象p的三個數(shù)據(jù)成員都是一個空的字符串。
二、執(zhí)行調(diào)用的構造函數(shù)的函數(shù)體語句,完成對數(shù)據(jù)成員的賦值,以此達到我們期望的創(chuàng)建一個指定Person對象,而不是空對象。
從上面的第二步就可以看到,我們在對三個const對象進行賦值操作,這顯然是不允許的操作,因此利用這個構造函數(shù)創(chuàng)建Person將以失敗告終。要想成功的創(chuàng)建一個特定的Person對象,我們需要構造函數(shù)初始化列表:
Person(string name, string phone, string addr)
:m_name(name), m_phone(phone), m_addr(addr){ } //冒號開始定義初始化列表
使用初始化列表創(chuàng)建對象的構造函數(shù)同樣是通過上述的兩個步驟來完成的,不同之處在于創(chuàng)建對象的數(shù)據(jù)成員時使用的不是默認構造函數(shù),而是根據(jù)指定參數(shù)調(diào)用了相應的構造函數(shù),以此創(chuàng)建特定的對象,而不是空對象。這樣一來,對象的數(shù)據(jù)成員的特定值在創(chuàng)建對象的時候就被賦予了相應的成員,而不是在創(chuàng)建對象完成之后再通過賦值語句去修改數(shù)據(jù)成員,因此利用構造函數(shù)初始化列表就可以成功的創(chuàng)建具有const數(shù)據(jù)成員的對對象了。
除了const的數(shù)據(jù)成員外,沒有默認構造函數(shù)的類類型或者是引用類型的成員,都必須在構造函數(shù)的初始化列表中進行初始化。
沒有默認構造函數(shù)的類類型成員,如果不在初始化列表中初始化的話,那么創(chuàng)建該對象的時候,由于沒有指定相應的“實參”,編譯器就會去調(diào)用默認構造函數(shù)來創(chuàng)建對象,必然會以失敗而告終。
引用類型的成員和const類型成員一樣,因為引用必須初始化,初始化后就不能修改,所以后期通過賦值來修改其值是錯誤的。
ps:數(shù)據(jù)成員被初始化的順序與構造函數(shù)初始化列表中的次序無關,而是與成員的定義順序一致。