當(dāng)前位置:首頁 > 芯聞號 > 充電吧
[導(dǎo)讀]在我們實際編程中,我們經(jīng)常會碰到變量初始化的問題,對于不同的變量初始化的手段多種多樣,比如說對于一個數(shù)組我們可以使用 int arr[] = {1,2,3}的方式初始化,又比如對于一個簡單的結(jié)構(gòu)體:[

在我們實際編程中,我們經(jīng)常會碰到變量初始化的問題,對于不同的變量初始化的手段多種多樣,比如說對于一個數(shù)組我們可以使用 int arr[] = {1,2,3}的方式初始化,又比如對于一個簡單的結(jié)構(gòu)體:


[cpp]view plaincopy structA { intx; inty; }a={1,2};

這些不同的初始化方法都有各自的適用范圍和作用,且對于類來說不能用這種初始化的方法,最主要的是沒有一種可以通用的初始化方法適用所有的場景,因此C++11中為了統(tǒng)一初始化方式,提出了列表初始化(list-initialization)的概念。

統(tǒng)一的初始化方法

在C++98/03中我們只能對普通數(shù)組和POD(plain old data,簡單來說就是可以用memcpy復(fù)制的對象)類型可以使用列表初始化,如下:


數(shù)組的初始化列表:


?int?arr[3]?=?{1,2,3}


POD類型的初始化列表:

[cpp]view plaincopy structA { intx; inty; }a={1,2}; 在C++11中初始化列表被適用性被放大,可以作用于任何類型對象的初始化。如下:



[cpp]view plaincopy classFoo { public: Foo(int){} private: Foo(constFoo&); }; int_tmain(intargc,_TCHAR*argv[]) { Fooa1(123);//調(diào)用Foo(int)構(gòu)造函數(shù)初始化 Fooa2=123;//errorFoo的拷貝構(gòu)造函數(shù)聲明為私有的,該處的初始化方式是隱式調(diào)用Foo(int)構(gòu)造函數(shù)生成一個臨時的匿名對象,再調(diào)用拷貝構(gòu)造函數(shù)完成初始化 Fooa3={123};//列表初始化 Fooa4{123};//列表初始化 inta5={3}; inta6{3}; return0; } 由上面的示例代碼可以看出,在C++11中,列表初始化不僅能完成對普通類型的初始化,還能完成對類的列表初始化,需要注意的是a3,a4都是列表初始化,私有的拷貝并不影響它,僅調(diào)用類的構(gòu)造函數(shù)而不需要拷貝構(gòu)造函數(shù),a4,a6的寫法是C++98/03所不具備的(可以不寫等號),是C++11新增的寫法。


同時列表初始化方法也適用于用new操作等圓括號進行初始化的地方,如下:


[cpp]view plaincopy int*a=newint{3}; doubleb=double{12.12}; int*arr=newint[]{1,2,3}; 讓人驚奇的是在C++11中可以使用列表初始化方法對堆中分配的數(shù)組進行初始化,而在C++98/03中是不能這樣做的。

列表初始化的一些細節(jié)

雖然列表初始化提供了統(tǒng)一的初始化方法,但是同時也會帶來一些使用上的疑惑需要各位苦逼碼農(nóng)注意,比如對下面的自定義類型的例子:



[cpp]view plaincopy structA { intx; inty; }a={123,321}; //a.x=123a.y=321 structB { intx; inty; B(int,int):x(0),y(0){} }b={123,321}; //b.x=0b.y=0 對于自定義的結(jié)構(gòu)體A來說也是普通的POD類型,使用列表初始化并不會引起問題,x,y都被正確的初始化了,但看下結(jié)構(gòu)體B和結(jié)構(gòu)體A的區(qū)別在于結(jié)構(gòu)體B定義了一個構(gòu)造函數(shù),并使用了成員初始化列表來初始化B的兩個變量,因此列表初始化在這里就不起作用了,B采用的是構(gòu)造函數(shù)的方式來完成變量的初始化工作。


那么如何區(qū)分一個類(class struct union)是否可以使用列表初始化來完成初始化工作呢?關(guān)鍵問題看這個類是否是一個聚合體(aggregate),首先看下C++中關(guān)于類是否是一個聚合體的定義:

(1)無用戶自定義構(gòu)造函數(shù)。

(2)無私有或者受保護的非靜態(tài)數(shù)據(jù)成員

(3)無基類

(4)無虛函數(shù)

(5)無{}和=直接初始化的非靜態(tài)數(shù)據(jù)成員。下面我們逐個對上述進行分析。

1、首先存在用戶自定義的構(gòu)造函數(shù)的情況,示例如下:


[cpp]view plaincopy structFoo { intx; inty; Foo(int,int){cout<<"Fooconstruction";} }; int_tmain(intargc,_TCHAR*argv[]) { Foofoo{123,321}; cout<<foo.x<<""<<foo.y; return0; } 輸出結(jié)果為:Fooconstruction -858993460 -858993460


可以看出對于有用戶自定義構(gòu)造函數(shù)的類使用初始化列表其成員初始化后變量值是一個隨機值,因此用戶必須以用戶自定義構(gòu)造函數(shù)來構(gòu)造對象。

2、類包含有私有的或者受保護的非靜態(tài)數(shù)據(jù)成員的情況

[cpp]view plaincopy structFoo { intx; inty; //Foo(int,int,double){} protected: doublez; }; int_tmain(intargc,_TCHAR*argv[]) { Foofoo{123,456,789.0}; cout<<foo.x<<""<<foo.y; return0; } 實例中z是一個受保護的成員變量,該程序直接在VS2013下編譯出錯,error C2440: 'initializing' : cannot convert from 'initializer-list' to 'Foo',而如果將z變量聲明為static則,可以用列表初始化來,示例:



[cpp]view plaincopy structFoo { intx; inty; //Foo(int,int,double){} protected: staticdoublez; }; int_tmain(intargc,_TCHAR*argv[]) { Foofoo{123,456};//如果寫成Foo foo{123,456,789.0}會提示error C2078: 初始值設(shè)定項太多 cout<<foo.x<<""<<foo.y; return0; } 程序輸出:123 456,因此可知靜態(tài)數(shù)據(jù)成員的初始化是不能通過初始化列表來完成初始化的,它的初始化還是遵循以往的靜態(tài)成員的初始化方式。


3、類含有基類或者虛函數(shù)

[cpp]view plaincopy structFoo { intx; inty; virtualvoidfunc(){}; }; int_tmain(intargc,_TCHAR*argv[]) { Foofoo{123,456}; cout<<foo.x<<""<<foo.y; return0; } 上例中類Foo中包含了一個虛函數(shù),該程序也是非法的,編譯不過的,錯誤信息和上述一樣cannot convert from 'initializer-list' to 'Foo'。
[cpp]view plaincopy structbase{}; structFoo:base { intx; inty; }; int_tmain(intargc,_TCHAR*argv[]) { Foofoo{123,456}; cout<<foo.x<<""<<foo.y; return0; } 上例中則是有基類的情況,類Foo從base中繼承,然后對Foo使用列表初始化,該程序也一樣無法通過編譯,錯誤信息仍然為cannot convert from 'initializer-list' to 'Foo',


4、類中不能有{}或者=直接初始化的費靜態(tài)數(shù)據(jù)成員


[cpp]view plaincopy structFoo { intx; inty=5; }; int_tmain(intargc,_TCHAR*argv[]) { Foofoo{123,456}; cout<<foo.x<<""<<foo.y; return0; } 在結(jié)構(gòu)體Foo中變量y直接用=進行初始化了,因此上述例子也不能使用列表初始化方法,需要注意的是在C++98/03中,類似于變量y這種直接用=進行初始化的方法是不允許的,但是在C++11中放寬了,是可以直接進行初始化的,對于一個類來說如果它的非靜態(tài)數(shù)據(jù)成員使用了=或者{}在聲明同時進行了初始化,那么它就不再是聚合類型了,不適合使用列表初始化方法了。


在上述4種不再適合使用列表初始化的例子中,需要注意的是一個類聲明了自己的構(gòu)造函數(shù)的情形,在這種情況下使用初始化列表是編譯器是不會給你報錯的,操作系統(tǒng)會給變量一個隨機的值,這種問題在代碼出BUG后是很難查找到的,因此這種情況不適合使用列表初始化需要特別注意,而其他不適合使用的情況編譯器會直接報錯,提醒你這些場景下使用列表初始化時不合法的。

那么是否有一種方法可以使得在類不是聚合類型的時候可以使用列表初始化方法呢?相信你肯定猜到了,作為一種很強大的語言不應(yīng)該也不會存在使用上的限制。自定義構(gòu)造函數(shù)+成員初始化列表的方式解決了上述類是非聚合類型使用列表初始化的限制??聪旅娴睦樱?/p>


[cpp]view plaincopy structFoo { intx; inty=5; virtualvoidfunc(){} private: intz; public: Foo(inti,intj,intk):x(i),y(j),z(k){cout<<z<<endl;} }; int_tmain(intargc,_TCHAR*argv[]) { Foofoo{123,456,789}; cout<<foo.x<<""<<foo.y; return0; } 輸出結(jié)果為 789 123 456 ,可見,盡管Foo中包含了私有的非靜態(tài)數(shù)據(jù)以及虛函數(shù),用戶自定義構(gòu)造函數(shù),并且使用成員列表初始化方法可以使得非聚合類型的類也可以使用列表初始化方法,因此在這里給各位看官提個建議,在對類的數(shù)據(jù)成員進行初始化的時候盡量在類的構(gòu)造函數(shù)中用成員初始化列表的方式來對數(shù)據(jù)成員進行初始化,這樣可以防止一些意外的錯誤。
初始化列表

在上面的使得一個類成為非聚合類的例子2、3、4中,這些非法的用法編譯器都報出的錯誤是cannot convert from 'initializer-list' to 'Foo',那么這個initializer-list是什么呢?為什么使用列表初始化方法是將initializer-list轉(zhuǎn)換成對應(yīng)的類類型呢?下面我們就來看看這個神秘的東西

1.任何長度的初始化列表

在C++11中,對于任意的STL容器都與未指定長度的數(shù)組有一樣的初始化能力,如: [cpp]view plaincopy intarr[]={1,2,3,4,5}; std::map

本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫毥谦F公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運行,同時企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險,如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機 衛(wèi)星通信

要點: 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運營商 數(shù)字經(jīng)濟

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學(xué)會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團)股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉