我們定義一個(gè)如下的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)這個(gè)類的第二個(gè)帶參數(shù)的構(gòu)造函數(shù)是錯(cuò)誤的。我們創(chuàng)建一個(gè)Person對(duì)象:
Person p("marcky", "13233232", "cqupt"); //調(diào)用帶參數(shù)的構(gòu)造函數(shù)創(chuàng)建一個(gè)Person對(duì)象
創(chuàng)建對(duì)象的過(guò)程分為了兩步:
一、從內(nèi)存中分配實(shí)際的空間給對(duì)象p,其三個(gè)字符串對(duì)象的數(shù)據(jù)成員是調(diào)用的默認(rèn)構(gòu)造函數(shù)初始化為空。也就說(shuō),此時(shí)為止,對(duì)象p的三個(gè)數(shù)據(jù)成員都是一個(gè)空的字符串。
二、執(zhí)行調(diào)用的構(gòu)造函數(shù)的函數(shù)體語(yǔ)句,完成對(duì)數(shù)據(jù)成員的賦值,以此達(dá)到我們期望的創(chuàng)建一個(gè)指定Person對(duì)象,而不是空對(duì)象。
從上面的第二步就可以看到,我們?cè)趯?duì)三個(gè)const對(duì)象進(jìn)行賦值操作,這顯然是不允許的操作,因此利用這個(gè)構(gòu)造函數(shù)創(chuàng)建Person將以失敗告終。要想成功的創(chuàng)建一個(gè)特定的Person對(duì)象,我們需要構(gòu)造函數(shù)初始化列表:
Person(string name, string phone, string addr)
:m_name(name), m_phone(phone), m_addr(addr){ } //冒號(hào)開(kāi)始定義初始化列表
使用初始化列表創(chuàng)建對(duì)象的構(gòu)造函數(shù)同樣是通過(guò)上述的兩個(gè)步驟來(lái)完成的,不同之處在于創(chuàng)建對(duì)象的數(shù)據(jù)成員時(shí)使用的不是默認(rèn)構(gòu)造函數(shù),而是根據(jù)指定參數(shù)調(diào)用了相應(yīng)的構(gòu)造函數(shù),以此創(chuàng)建特定的對(duì)象,而不是空對(duì)象。這樣一來(lái),對(duì)象的數(shù)據(jù)成員的特定值在創(chuàng)建對(duì)象的時(shí)候就被賦予了相應(yīng)的成員,而不是在創(chuàng)建對(duì)象完成之后再通過(guò)賦值語(yǔ)句去修改數(shù)據(jù)成員,因此利用構(gòu)造函數(shù)初始化列表就可以成功的創(chuàng)建具有const數(shù)據(jù)成員的對(duì)對(duì)象了。
除了const的數(shù)據(jù)成員外,沒(méi)有默認(rèn)構(gòu)造函數(shù)的類類型或者是引用類型的成員,都必須在構(gòu)造函數(shù)的初始化列表中進(jìn)行初始化。
沒(méi)有默認(rèn)構(gòu)造函數(shù)的類類型成員,如果不在初始化列表中初始化的話,那么創(chuàng)建該對(duì)象的時(shí)候,由于沒(méi)有指定相應(yīng)的“實(shí)參”,編譯器就會(huì)去調(diào)用默認(rèn)構(gòu)造函數(shù)來(lái)創(chuàng)建對(duì)象,必然會(huì)以失敗而告終。
引用類型的成員和const類型成員一樣,因?yàn)橐帽仨毘跏蓟跏蓟缶筒荒苄薷模院笃谕ㄟ^(guò)賦值來(lái)修改其值是錯(cuò)誤的。
ps:數(shù)據(jù)成員被初始化的順序與構(gòu)造函數(shù)初始化列表中的次序無(wú)關(guān),而是與成員的定義順序一致。