溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

對(duì)象的構(gòu)造順序(十六)

發(fā)布時(shí)間:2020-07-21 10:51:11 來(lái)源:網(wǎng)絡(luò) 閱讀:1175 作者:上帝之子521 欄目:編程語(yǔ)言

        在 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)造順序(十六)

        我們看到局部對(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)編譯看看

對(duì)象的構(gòu)造順序(十六)

        編譯直接出錯(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í)行的

對(duì)象的構(gòu)造順序(十六)

        確實(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é)果

對(duì)象的構(gòu)造順序(十六)

        這個(gè)結(jié)果貌似跟我們指定編譯的順序有關(guān)系,我們?cè)賮?lái)看看BCC編譯器呢

對(duì)象的構(gòu)造順序(十六)

        再來(lái)試試 VS2010

對(duì)象的構(gòu)造順序(十六)

        以前博主在書(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。

向AI問(wèn)一下細(xì)節(jié)

免責(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)容。

AI