轉(zhuǎn)載:https://blog.csdn.net/lws123253/article/details/80368047
定義
與其他函數(shù)不同皇耗,構(gòu)造函數(shù)除了有名字棚点,參數(shù)列表和函數(shù)體之外装哆,還可以有初始化列表朱庆,初始化列表以冒號(hào)開(kāi)頭相恃,后跟一系列以逗號(hào)分隔的初始化字段钝满。
class foo
{
public:
foo(string s, int i):name(s), id(i){} ; // 初始化列表
private:
string name ;int id ;
};
從概念上來(lái)講畦木,構(gòu)造函數(shù)的執(zhí)行可以分成兩個(gè)階段袖扛,初始化階段和計(jì)算階段,初始化階段先于計(jì)算階段.
初始化階段
所有類(lèi)類(lèi)型(class type)的成員都會(huì)在初始化階段初始化十籍,即使該成員沒(méi)有出現(xiàn)在構(gòu)造函數(shù)的初始化列表中.
計(jì)算階段
一般用于執(zhí)行構(gòu)造函數(shù)體內(nèi)的賦值操作蛆封。
下面的代碼中,其中Test1有構(gòu)造函數(shù)勾栗,拷貝構(gòu)造函數(shù)及賦值運(yùn)算符惨篱,為的是方便查看結(jié)果,Test2是個(gè)測(cè)試類(lèi)围俘,它以Test1的對(duì)象為成員砸讳,我們看一下Test2的構(gòu)造函數(shù)是怎么樣執(zhí)行的。
class Test1
{
public:
Test1() // 無(wú)參構(gòu)造函數(shù)
{cout << "Construct Test1" << endl ;}
Test1(const Test1& t1) // 拷貝構(gòu)造函數(shù)
{cout << "Copy constructor for Test1" << endl ;this->a = t1.a ;}
Test1& operator = (const Test1& t1) // 賦值運(yùn)算符
{cout << "assignment for Test1" << endl ;this->a = t1.a ;return *this;}
int a ;
};
class Test2
{
public:
Test1 test1 ;
Test2(Test1 &t1)
{test1 = t1 ;}
};
調(diào)用代碼:
Test1 t1 ;
Test2 t2(t1) ;
輸出:
Construct Test1
Construct Test1
assignment for Test1
解釋一下:
第一行輸出對(duì)應(yīng)調(diào)用代碼中第一行界牡,構(gòu)造一個(gè)Test1對(duì)象
第二行輸出對(duì)應(yīng)Test2構(gòu)造函數(shù)中的代碼簿寂,用默認(rèn)的構(gòu)造函數(shù)初始化對(duì)象test1 // 這就是所謂的初始化階段
第三行輸出對(duì)應(yīng)Test2的賦值運(yùn)算符,對(duì)test1執(zhí)行賦值操作 // 這就是所謂的計(jì)算階段
使用初始化列表的原因
初始化類(lèi)的成員有兩種方式:
1 是使用初始化列表宿亡,
2 是在構(gòu)造函數(shù)體內(nèi)進(jìn)行賦值操作常遂。
主要是性能問(wèn)題,對(duì)于內(nèi)置類(lèi)型挽荠,如int, float等烈钞,使用初始化類(lèi)表和在構(gòu)造函數(shù)體內(nèi)初始化差別不是很大,但是對(duì)于類(lèi)類(lèi)型來(lái)說(shuō)坤按,最好使用初始化列表毯欣,為什么呢?由下面的測(cè)試可知臭脓,使用初始化列表少了一次調(diào)用默認(rèn)構(gòu)造函數(shù)的過(guò)程酗钞,這對(duì)于數(shù)據(jù)密集型的類(lèi)來(lái)說(shuō),是非常高效的。同樣看上面的例子砚作,我們使用初始化列表來(lái)實(shí)現(xiàn)Test2的構(gòu)造函數(shù)窘奏。
class Test2
{
public:
Test1 test1 ;
Test2(Test1 &t1):test1(t1){}
}
使用同樣的調(diào)用代碼,輸出結(jié)果如下:
Construct Test1
Copy constructor for Test1
第一行輸出對(duì)應(yīng) 調(diào)用代碼的第一行
第二行輸出對(duì)應(yīng)Test2的初始化列表葫录,直接調(diào)用拷貝構(gòu)造函數(shù)初始化test1着裹,省去了調(diào)用默認(rèn)構(gòu)造函數(shù)的過(guò)程。
所以一個(gè)好的原則是米同,能使用初始化列表的時(shí)候盡量使用初始化列表.
必須使用初始化列表的時(shí)候
除了性能問(wèn)題之外骇扇,有些時(shí)候合初始化列表是不可或缺的.
以下幾種情況時(shí)必須使用初始化列表:
1.常量成員,因?yàn)槌A恐荒艹跏蓟荒苜x值面粮,所以必須放在初始化列表里面
2.引用類(lèi)型少孝,引用必須在定義的時(shí)候初始化,并且不能重新賦值熬苍,所以也要寫(xiě)在初始化列表里面
3.沒(méi)有默認(rèn)構(gòu)造函數(shù)的類(lèi)類(lèi)型稍走,因?yàn)槭褂贸跏蓟斜砜梢圆槐卣{(diào)用默認(rèn)構(gòu)造函數(shù)來(lái)初始化,而是直接調(diào)用拷貝構(gòu)造函數(shù)初始化
class Test1
{
public:
Test1(int a):i(a){}
int i;
};
class Test2
{
public:
Test1 test1 ;
Test2(Test1 &t1)
{test1 = t1 ;}
};
以上代碼無(wú)法通過(guò)編譯柴底,因?yàn)門(mén)est2的構(gòu)造函數(shù)中test1 = t1這一行實(shí)際上分成兩步執(zhí)行:
- 調(diào)用Test1的默認(rèn)構(gòu)造函數(shù)來(lái)初始化test1
由于Test1沒(méi)有默認(rèn)的構(gòu)造函數(shù)婿脸,所以1 無(wú)法執(zhí)行,故而編譯錯(cuò)誤柄驻。正確的代碼如下盖淡,使用初始化列表代替賦值操作
class Test2
{
public:
Test1 test1 ;
Test2(int x):test1(x){}
}
成員變量的順序
成員是按照他們?cè)陬?lèi)中出現(xiàn)的順序進(jìn)行初始化的,而不是按照他們?cè)诔跏蓟斜沓霈F(xiàn)的順序初始化的凿歼,看代碼:
class foo
{
public:
int i ;int j ;
foo(int x):i(x), j(i){}; // ok, 先初始化i褪迟,后初始化j
};
再看下面的代碼:
class foo
{
public:
int i ;int j ;
foo(int x):j(x), i(j){} // i值未定義
};
這里i的值是未定義的因?yàn)殡m然j在初始化列表里面出現(xiàn)在i前面,但是i先于j定義答憔,所以先初始化i味赃,而i由j初始化,此時(shí)j尚未初始化虐拓,所以導(dǎo)致i的值未定義心俗。一個(gè)好的習(xí)慣是,按照成員定義的順序進(jìn)行初始化蓉驹。