您好,登錄后才能下訂單哦!
在 C++ 中的類(lèi)可以定義多個(gè)對(duì)象,那么對(duì)象構(gòu)造的順序是怎樣的呢?對(duì)于局部對(duì)象:當(dāng)程序執(zhí)行流到達(dá)對(duì)象的定義語(yǔ)句時(shí)進(jìn)行構(gòu)造。我們以代碼為例進(jìn)行分析
#include <stdio.h> class Test { private: int mi; public: Test(int i) { mi = i; printf("Test(int i): %d\n", mi); } Test(const Test& obj) { mi = obj.mi; printf("Test(const Test& obj): %d\n", mi); } }; int main() { int i = 0; Test a1 = i; // Test(int i): 0 while( i < 3 ) { Test a2 = ++i; // Test(int i): 1, 2, 3 } if( i < 4 ) { Test a = a1; // Test(const Test& obj): 0 } else { Test a(100); } return 0; }
我們按照程序的執(zhí)行流可以看到先是執(zhí)行對(duì)象 a1 的創(chuàng)建,接著是對(duì)象 a2 的創(chuàng)建 3 次,最后是對(duì)象 a 的拷貝構(gòu)造。我們看看結(jié)果是否如我們所分析的那樣
我們看到局部對(duì)象的構(gòu)造順序確實(shí)如我們所想的那樣。如果我們使用 goto 語(yǔ)句呢,我們看個(gè)代碼
#include <stdio.h> class Test { private: int mi; public: Test(int i) { mi = i; printf("Test(int i): %d\n", mi); } Test(const Test& obj) { mi = obj.mi; printf("Test(const Test& obj): %d\n", mi); } int getMI() { return mi; } }; int main() { int i = 0; Test a1 = i; // Test(int i): 0 while( i < 3 ) { Test a2 = ++i; // Test(int i): 1, 2, 3 } goto End; Test a(100); End: printf("a.mi = %d\n", a.getMI()); return 0; }
我們來(lái)編譯看看
編譯直接出錯(cuò),因?yàn)槲覀兪褂昧?goto 語(yǔ)句,導(dǎo)致程序的執(zhí)行流出錯(cuò)了。
接下來(lái)我們來(lái)看看堆對(duì)象的構(gòu)造順序,當(dāng)程序執(zhí)行流到達(dá) new 語(yǔ)句時(shí)創(chuàng)建對(duì)象,使用 new 創(chuàng)建對(duì)象將自動(dòng)觸發(fā)構(gòu)造函數(shù)的調(diào)用。
下來(lái)還是以代碼為例來(lái)分析堆對(duì)象的構(gòu)造順序
#include <stdio.h> class Test { private: int mi; public: Test(int i) { mi = i; printf("Test(int i): %d\n", mi); } Test(const Test& obj) { mi = obj.mi; printf("Test(const Test& obj): %d\n", mi); } int getMI() { return mi; } }; int main() { int i = 0; Test* a1 = new Test(i); // Test(int i): 0 while( ++i < 10 ) if( i % 2 ) new Test(i); // Test(int i): 1, 3, 5, 7, 9 if( i < 4 ) new Test(*a1); else new Test(100); // Test(int i): 100 return 0; }
我們看看是否如我們所注釋的那樣執(zhí)行的
確實(shí),堆對(duì)象的構(gòu)造順序是跟 new 關(guān)鍵字有關(guān)系的。下來(lái)我們來(lái)看看全局對(duì)象,對(duì)象的構(gòu)造順序是不確定的,不同的編譯器使用不同的規(guī)則來(lái)確定構(gòu)造順序。還是以代碼為例來(lái)進(jìn)行驗(yàn)證
test.h 源碼
#ifndef _TEST_H_ #define _TEST_H_ #include <stdio.h> class Test { public: Test(const char* s) { printf("%s\n", s); } }; #endif
t1.cpp 源碼
#include "test.h" Test t1("t1");
t2.cpp 源碼
#include "test.h" Test t2("t2");
t3.cpp 源碼
#include "test.h" Test t3("t3");
test.cpp 源碼
#include "test.h" Test t4("t4"); int main() { Test t5("t5"); return 0; }
我們來(lái)編譯看看結(jié)果
這個(gè)結(jié)果貌似跟我們指定編譯的順序有關(guān)系,我們?cè)賮?lái)看看BCC編譯器呢
再來(lái)試試 VS2010
以前博主在書(shū)上和視頻中看到過(guò)全局對(duì)象的構(gòu)造順序是不確定的,可能現(xiàn)在的編譯器做了優(yōu)化吧。反正我們記住就可以了,盡量避免使用全局對(duì)象。通過(guò)對(duì)對(duì)象的構(gòu)造順序的學(xué)習(xí),總稽核如下:局部對(duì)象的構(gòu)造順序依賴(lài)于程序的執(zhí)行流;堆對(duì)象的構(gòu)造順序依賴(lài)于 new 的使用順序;全局對(duì)象的構(gòu)造順序是不確定的
歡迎大家一起來(lái)學(xué)習(xí) C++ 語(yǔ)言,可以加我QQ:243343083。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。