C++初始化列表學(xué)習(xí)
何謂初始化列表
與其他函數(shù)不同,構(gòu)造函數(shù)除了有名字,參數(shù)列表和函數(shù)體之外,還可以有初始化列表,初始化列表以冒號(hào)開(kāi)頭,后跟一系列以逗號(hào)分隔的初始化字段。在C++中,struct和class的唯一區(qū)別是默認(rèn)的克訪問(wèn)性不同,而這里我們不考慮訪問(wèn)性的問(wèn)題,所以下面的代碼都以struct來(lái)演示。
struct foo
{
string name ;
int id ;
foo(string s, int i):name(s), id(i){} ; // 初始化列表
};
構(gòu)造函數(shù)的兩個(gè)執(zhí)行階段
構(gòu)造函數(shù)的執(zhí)行可以分成兩個(gè)階段,初始化階段和計(jì)算階段,初始化階段先于計(jì)算階段。
初始化階段
所有類類型(class type)的成員都會(huì)在初始化階段初始化,即使該成員沒(méi)有出現(xiàn)在構(gòu)造函數(shù)的初始化列表中。
計(jì)算階段
一般用于執(zhí)行構(gòu)造函數(shù)體內(nèi)的賦值操作,下面的代碼定義兩個(gè)結(jié)構(gòu)體,其中Test1有構(gòu)造函數(shù),拷貝構(gòu)造函數(shù)及賦值運(yùn)算符,為的是方便查看結(jié)果。Test2是個(gè)測(cè)試類,它以Test1的對(duì)象為成員,我們看一下Test2的構(gòu)造函數(shù)是怎么樣執(zhí)行的。
struct Test1
{
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 ;
};
struct Test2
{
Test1 test1 ;
Test2(Test1 &t1)
{
test1 = t1 ;
}
};
調(diào)用代碼
Test1 t1 ;
Test2 t2(t1) ;
輸出
解釋一下,第一行輸出對(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)Test1的賦值運(yùn)算符,對(duì)test1執(zhí)行賦值操作,這就是所謂的計(jì)算階段。
為什么使用初始化列表
初始化類的成員有兩種方式,一是使用初始化列表,二是在構(gòu)造函數(shù)體內(nèi)進(jìn)行賦值操作。使用初始化列表主要是基于性能問(wèn)題,對(duì)于內(nèi)置類型,如int, float等,使用初始化類表和在構(gòu)造函數(shù)體內(nèi)初始化差別不是很大,但是對(duì)于類類型來(lái)說(shuō),最好使用初始化列表,為什么呢?由上面的測(cè)試可知,使用初始化列表少了一次調(diào)用默認(rèn)構(gòu)造函數(shù)的過(guò)程,這對(duì)于數(shù)據(jù)密集型的類來(lái)說(shuō),是非常高效的。同樣看上面的例子,我們使用初始化列表來(lái)實(shí)現(xiàn)Test2的構(gòu)造函數(shù)
struct Test2
{
Test1 test1 ;
Test2(Test1 &t1):test1(t1){}
}
使用同樣的調(diào)用代碼,輸出結(jié)果如下。
第一行輸出對(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í)候盡量使用初始化列表。
哪些東西必須放在初始化列表中
除了性能問(wèn)題之外,有些時(shí)場(chǎng)合初始化列表是不可或缺的,以下幾種情況時(shí)必須使用初始化列表
常量成員,因?yàn)槌A恐荒艹跏蓟荒苜x值,所以必須放在初始化列表里面
引用類型,引用必須在定義的時(shí)候初始化,并且不能重新賦值,所以也要寫(xiě)在初始化列表里面
沒(méi)有默認(rèn)構(gòu)造函數(shù)的類類型,因?yàn)槭褂贸跏蓟斜砜梢圆槐卣{(diào)用默認(rèn)構(gòu)造函數(shù)來(lái)初始化,而是直接調(diào)用拷貝構(gòu)造函數(shù)初始化。
對(duì)于沒(méi)有默認(rèn)構(gòu)造函數(shù)的類,我們看一個(gè)例子。
struct Test1
{
Test1(int a):i(a){}
int i ;
};
struct Test2
{
Test1 test1 ;
Test2(Test1 &t1)
{
test1 = t1 ;
}
};
以上代碼無(wú)法通過(guò)編譯,因?yàn)門(mén)est2的構(gòu)造函數(shù)中test1 = t1這一行實(shí)際上分成兩步執(zhí)行。
1. 調(diào)用Test1的默認(rèn)構(gòu)造函數(shù)來(lái)初始化test1
2. 調(diào)用Test1的賦值運(yùn)算符給test1賦值
但是由于Test1沒(méi)有默認(rèn)的構(gòu)造函數(shù),所謂第一步無(wú)法執(zhí)行,故而編譯錯(cuò)誤。正確的代碼如下,使用初始化列表代替賦值操作。
struct Test2
{
Test1 test1 ;
Test2(Test1 &t1):test1(t1){}
}
成員變量的初始化順序
成員是按照他們?cè)陬愔谐霈F(xiàn)的順序進(jìn)行初始化的,而不是按照他們?cè)诔跏蓟斜沓霈F(xiàn)的順序初始化的,看代碼。
struct foo
{
int i ;
int j ;
foo(int x):i(x), j(i){}; // ok, 先初始化i,后初始化j
};
再看下面的代碼
struct foo
{
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)行初始化。
上一篇:如何通過(guò)函數(shù)指針調(diào)用函數(shù)(實(shí)現(xiàn)代碼)
欄 目:C語(yǔ)言
下一篇:C++的sstream標(biāo)準(zhǔn)庫(kù)詳細(xì)介紹
本文標(biāo)題:C++初始化列表學(xué)習(xí)
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/4199.html
您可能感興趣的文章
- 04-02c語(yǔ)言沒(méi)有round函數(shù) round c語(yǔ)言
- 01-10深入理解C++中常見(jiàn)的關(guān)鍵字含義
- 01-10使用C++實(shí)現(xiàn)全排列算法的方法詳解
- 01-10c++中inline的用法分析
- 01-10用C++實(shí)現(xiàn)DBSCAN聚類算法
- 01-10全排列算法的非遞歸實(shí)現(xiàn)與遞歸實(shí)現(xiàn)的方法(C++)
- 01-10C++大數(shù)模板(推薦)
- 01-10淺談C/C++中的static與extern關(guān)鍵字的使用詳解
- 01-10深入C/C++浮點(diǎn)數(shù)在內(nèi)存中的存儲(chǔ)方式詳解
- 01-10深入理解C/C++混合編程


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹(shù)的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02c語(yǔ)言函數(shù)調(diào)用后清空內(nèi)存 c語(yǔ)言調(diào)用
- 04-02func函數(shù)+在C語(yǔ)言 func函數(shù)在c語(yǔ)言中
- 04-02c語(yǔ)言的正則匹配函數(shù) c語(yǔ)言正則表達(dá)
- 04-02c語(yǔ)言用函數(shù)寫(xiě)分段 用c語(yǔ)言表示分段
- 04-02c語(yǔ)言中對(duì)數(shù)函數(shù)的表達(dá)式 c語(yǔ)言中對(duì)
- 04-02c語(yǔ)言編寫(xiě)函數(shù)冒泡排序 c語(yǔ)言冒泡排
- 04-02c語(yǔ)言沒(méi)有round函數(shù) round c語(yǔ)言
- 04-02c語(yǔ)言分段函數(shù)怎么求 用c語(yǔ)言求分段
- 04-02C語(yǔ)言中怎么打出三角函數(shù) c語(yǔ)言中怎
- 04-02c語(yǔ)言調(diào)用函數(shù)求fibo C語(yǔ)言調(diào)用函數(shù)求
隨機(jī)閱讀
- 01-10delphi制作wav文件的方法
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 01-11Mac OSX 打開(kāi)原生自帶讀寫(xiě)NTFS功能(圖文
- 04-02jquery與jsp,用jquery
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 01-10C#中split用法實(shí)例總結(jié)
- 01-10SublimeText編譯C開(kāi)發(fā)環(huán)境設(shè)置