淺析C++中的C++構(gòu)造函數(shù)
在下面文章主要介紹C++構(gòu)造函數(shù)的相關(guān)問(wèn)題,使用一個(gè)簡(jiǎn)單的構(gòu)造函數(shù)定義了名為 Taxi 的類,然后使用 new 運(yùn)算符來(lái)實(shí)例化該類,在為新對(duì)象分配內(nèi)存之后,new 運(yùn)算符立即調(diào)用 Taxi 構(gòu)造函數(shù)。
C++構(gòu)造函數(shù)初始化列表以一個(gè)冒號(hào)開始,接著是以逗號(hào)分隔的數(shù)據(jù)成員列表,每個(gè)數(shù)據(jù)成員后面跟一個(gè)放在括號(hào)中的初始化式。例如:
- class CExample {
- public:
- int a;
- float b;
- //構(gòu)造函數(shù)初始化列表
- CExample(): a(0),b(8.8)
- {}
- //構(gòu)造函數(shù)內(nèi)部賦值
- CExample()
- {
- a=0;
- b=8.8;
- }
- };
上面的例子中兩個(gè)構(gòu)造函數(shù)的結(jié)果是一樣的。上面的C++構(gòu)造函數(shù)(使用初始化列表的構(gòu)造函數(shù))顯式的初始化類的成員;而沒(méi)使用初始化列表的構(gòu)造函數(shù)是對(duì)類的成員賦值,并沒(méi)有進(jìn)行顯式的初始化。
初始化和賦值對(duì)內(nèi)置類型的成員沒(méi)有什么大的區(qū)別,像上面的任一個(gè)構(gòu)造函數(shù)都可以。對(duì)非內(nèi)置類型成員變量,為了避免兩次構(gòu)造,推薦使用類構(gòu)造函數(shù)初始化列表。但有的時(shí)候必須用帶有初始化列表的構(gòu)造函數(shù):
- class CMyClass {
- CMyClass(int x, int y);
- int m_x;
- int m_y;
- };
- CMyClass::CMyClass(int x, int y) : m_y(y), m_x(m_y)
- {
- }
成員類型是沒(méi)有默認(rèn)C++構(gòu)造函數(shù)的類。若沒(méi)有提供顯示初始化式,則編譯器隱式使用成員類型的默認(rèn)C++構(gòu)造函數(shù),若類沒(méi)有默認(rèn)構(gòu)造函數(shù),則編譯器嘗試使用默認(rèn)構(gòu)造函數(shù)將會(huì)失敗。
結(jié)果上相同,但是性能上存在很大的差別。因?yàn)轭愵愋偷臄?shù)據(jù)成員對(duì)象在進(jìn)入函數(shù)體前已經(jīng)構(gòu)造完成,也就是說(shuō)在成員初始化列表處進(jìn)行構(gòu)造對(duì)象的工作。調(diào)用構(gòu)造函數(shù),在進(jìn)入函數(shù)體之后,進(jìn)行的是對(duì)已經(jīng)構(gòu)造好的類對(duì)象的賦值,又調(diào)用個(gè)拷貝賦值操作符才能完成(如果并未提供,則使用編譯器提供的默認(rèn)按成員賦值行為。
你可能以為上面的代碼將會(huì)首先做m_y=I,然后做m_x=m_y,最后它們有相同的值。但是編譯器先初始化m_x,然后是m_y,,因?yàn)樗鼈兪前催@樣的順序聲明的。結(jié)果是m_x將有一個(gè)不可預(yù)測(cè)的值。
有兩種方法避免它,一個(gè)是總是按照你希望它們被初始化的順序聲明成員,第二個(gè)是,如果你決定使用初始化列表,總是按照它們聲明的順序羅列這些成員。這將有助于消除混淆。
【編輯推薦】