www.久久久久|狼友网站av天堂|精品国产无码a片|一级av色欲av|91在线播放视频|亚洲无码主播在线|国产精品草久在线|明星AV网站在线|污污内射久久一区|婷婷综合视频网站

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

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


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

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

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

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


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


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


POD類型的初始化列表:

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



[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ù)生成一個(gè)臨時(shí)的匿名對(duì)象,再調(diào)用拷貝構(gòu)造函數(shù)完成初始化 Fooa3={123};//列表初始化 Fooa4{123};//列表初始化 inta5={3}; inta6{3}; return0; } 由上面的示例代碼可以看出,在C++11中,列表初始化不僅能完成對(duì)普通類型的初始化,還能完成對(duì)類的列表初始化,需要注意的是a3,a4都是列表初始化,私有的拷貝并不影響它,僅調(diào)用類的構(gòu)造函數(shù)而不需要拷貝構(gòu)造函數(shù),a4,a6的寫法是C++98/03所不具備的(可以不寫等號(hào)),是C++11新增的寫法。


同時(shí)列表初始化方法也適用于用new操作等圓括號(hào)進(jìn)行初始化的地方,如下:


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

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

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



[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 對(duì)于自定義的結(jié)構(gòu)體A來(lái)說(shuō)也是普通的POD類型,使用列表初始化并不會(huì)引起問(wèn)題,x,y都被正確的初始化了,但看下結(jié)構(gòu)體B和結(jié)構(gòu)體A的區(qū)別在于結(jié)構(gòu)體B定義了一個(gè)構(gòu)造函數(shù),并使用了成員初始化列表來(lái)初始化B的兩個(gè)變量,因此列表初始化在這里就不起作用了,B采用的是構(gòu)造函數(shù)的方式來(lái)完成變量的初始化工作。


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

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

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

(3)無(wú)基類

(4)無(wú)虛函數(shù)

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

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


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

2、類包含有私有的或者受保護(hù)的非靜態(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; } 實(shí)例中z是一個(gè)受保護(hù)的成員變量,該程序直接在VS2013下編譯出錯(cuò),error C2440: 'initializing' : cannot convert from 'initializer-list' to 'Foo',而如果將z變量聲明為static則,可以用列表初始化來(lái),示例:



[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}會(huì)提示error C2078: 初始值設(shè)定項(xiàng)太多 cout<<foo.x<<""<<foo.y; return0; } 程序輸出:123 456,因此可知靜態(tài)數(shù)據(jù)成員的初始化是不能通過(guò)初始化列表來(lái)完成初始化的,它的初始化還是遵循以往的靜態(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中包含了一個(gè)虛函數(shù),該程序也是非法的,編譯不過(guò)的,錯(cuò)誤信息和上述一樣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中繼承,然后對(duì)Foo使用列表初始化,該程序也一樣無(wú)法通過(guò)編譯,錯(cuò)誤信息仍然為cannot convert from 'initializer-list' to 'Foo',


4、類中不能有{}或者=直接初始化的費(fèi)靜態(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直接用=進(jìn)行初始化了,因此上述例子也不能使用列表初始化方法,需要注意的是在C++98/03中,類似于變量y這種直接用=進(jìn)行初始化的方法是不允許的,但是在C++11中放寬了,是可以直接進(jìn)行初始化的,對(duì)于一個(gè)類來(lái)說(shuō)如果它的非靜態(tài)數(shù)據(jù)成員使用了=或者{}在聲明同時(shí)進(jìn)行了初始化,那么它就不再是聚合類型了,不適合使用列表初始化方法了。


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

那么是否有一種方法可以使得在類不是聚合類型的時(shí)候可以使用列表初始化方法呢?相信你肯定猜到了,作為一種很強(qiáng)大的語(yǔ)言不應(yīng)該也不會(huì)存在使用上的限制。自定義構(gòu)造函數(shù)+成員初始化列表的方式解決了上述類是非聚合類型使用列表初始化的限制。看下面的例子:


[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ù),并且使用成員列表初始化方法可以使得非聚合類型的類也可以使用列表初始化方法,因此在這里給各位看官提個(gè)建議,在對(duì)類的數(shù)據(jù)成員進(jìn)行初始化的時(shí)候盡量在類的構(gòu)造函數(shù)中用成員初始化列表的方式來(lái)對(duì)數(shù)據(jù)成員進(jìn)行初始化,這樣可以防止一些意外的錯(cuò)誤。
初始化列表

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

1.任何長(zhǎng)度的初始化列表

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

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

LED驅(qū)動(dòng)電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: 驅(qū)動(dòng)電源

在工業(yè)自動(dòng)化蓬勃發(fā)展的當(dāng)下,工業(yè)電機(jī)作為核心動(dòng)力設(shè)備,其驅(qū)動(dòng)電源的性能直接關(guān)系到整個(gè)系統(tǒng)的穩(wěn)定性和可靠性。其中,反電動(dòng)勢(shì)抑制與過(guò)流保護(hù)是驅(qū)動(dòng)電源設(shè)計(jì)中至關(guān)重要的兩個(gè)環(huán)節(jié),集成化方案的設(shè)計(jì)成為提升電機(jī)驅(qū)動(dòng)性能的關(guān)鍵。

關(guān)鍵字: 工業(yè)電機(jī) 驅(qū)動(dòng)電源

LED 驅(qū)動(dòng)電源作為 LED 照明系統(tǒng)的 “心臟”,其穩(wěn)定性直接決定了整個(gè)照明設(shè)備的使用壽命。然而,在實(shí)際應(yīng)用中,LED 驅(qū)動(dòng)電源易損壞的問(wèn)題卻十分常見,不僅增加了維護(hù)成本,還影響了用戶體驗(yàn)。要解決這一問(wèn)題,需從設(shè)計(jì)、生...

關(guān)鍵字: 驅(qū)動(dòng)電源 照明系統(tǒng) 散熱

根據(jù)LED驅(qū)動(dòng)電源的公式,電感內(nèi)電流波動(dòng)大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

關(guān)鍵字: LED 設(shè)計(jì) 驅(qū)動(dòng)電源

電動(dòng)汽車(EV)作為新能源汽車的重要代表,正逐漸成為全球汽車產(chǎn)業(yè)的重要發(fā)展方向。電動(dòng)汽車的核心技術(shù)之一是電機(jī)驅(qū)動(dòng)控制系統(tǒng),而絕緣柵雙極型晶體管(IGBT)作為電機(jī)驅(qū)動(dòng)系統(tǒng)中的關(guān)鍵元件,其性能直接影響到電動(dòng)汽車的動(dòng)力性能和...

關(guān)鍵字: 電動(dòng)汽車 新能源 驅(qū)動(dòng)電源

在現(xiàn)代城市建設(shè)中,街道及停車場(chǎng)照明作為基礎(chǔ)設(shè)施的重要組成部分,其質(zhì)量和效率直接關(guān)系到城市的公共安全、居民生活質(zhì)量和能源利用效率。隨著科技的進(jìn)步,高亮度白光發(fā)光二極管(LED)因其獨(dú)特的優(yōu)勢(shì)逐漸取代傳統(tǒng)光源,成為大功率區(qū)域...

關(guān)鍵字: 發(fā)光二極管 驅(qū)動(dòng)電源 LED

LED通用照明設(shè)計(jì)工程師會(huì)遇到許多挑戰(zhàn),如功率密度、功率因數(shù)校正(PFC)、空間受限和可靠性等。

關(guān)鍵字: LED 驅(qū)動(dòng)電源 功率因數(shù)校正

在LED照明技術(shù)日益普及的今天,LED驅(qū)動(dòng)電源的電磁干擾(EMI)問(wèn)題成為了一個(gè)不可忽視的挑戰(zhàn)。電磁干擾不僅會(huì)影響LED燈具的正常工作,還可能對(duì)周圍電子設(shè)備造成不利影響,甚至引發(fā)系統(tǒng)故障。因此,采取有效的硬件措施來(lái)解決L...

關(guān)鍵字: LED照明技術(shù) 電磁干擾 驅(qū)動(dòng)電源

開關(guān)電源具有效率高的特性,而且開關(guān)電源的變壓器體積比串聯(lián)穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機(jī)重量也有所下降,所以,現(xiàn)在的LED驅(qū)動(dòng)電源

關(guān)鍵字: LED 驅(qū)動(dòng)電源 開關(guān)電源

LED驅(qū)動(dòng)電源是把電源供應(yīng)轉(zhuǎn)換為特定的電壓電流以驅(qū)動(dòng)LED發(fā)光的電壓轉(zhuǎn)換器,通常情況下:LED驅(qū)動(dòng)電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: LED 隧道燈 驅(qū)動(dòng)電源
關(guān)閉