亚洲免费在线-亚洲免费在线播放-亚洲免费在线观看-亚洲免费在线观看视频-亚洲免费在线看-亚洲免费在线视频

淺出C++對象模型——理解構造函數、析構函數執

系統 2739 0

本文主要說明對象創建時構造函數的執行順序,對象成員的初始化順序;對象銷毀時析構函數的執行順序,對象成員的銷毀順序。


“對象的構造從類層次的最根處開始,在每一層中,首先調用基類的構造函數,然后調用成員對象的構造函數。析構則嚴格按照與構造相反的次序執行,該次序是唯一的,否則編譯器將無法自動執行析構過程。

一個有趣的現象是,成員對象初始化的次序完全不受它們在初始化表中次序的影響,只由成員對象在類中聲明的次序決定。這是因為類的聲明是唯一的,而類的構造函數可以有多個,因此會有多個不同次序的初始化表。如果成員對象按照初始化表的次序進行構造,這將導致析構函數無法得到唯一的逆序。”(引用自 References[1]

從這里看,每種語言特性的存在必有其原因,學習這些特性就是理解這些特性存在的原因。

下面的一段代碼是對上面這段話的說明,其中有 4 個類 Foo,Bar,Base,Derived ,它們的構造函數、拷貝構造函數、析構函數都有信息輸出。

    #include <iostream>
using namespace std;

class Foo
{
public:
        Foo() { cout << "Foo constructor" << endl; }
        Foo(const Foo &foo) { cout << "Foo copy constructor" << endl; }
        ~Foo() { cout << "Foo deconstructor" << endl; }
};

class Bar
{
public:
        Bar() { cout << "Bar constructor" << endl; }
        Bar(const Bar &bar) { cout << "Bar copy constructor" << endl; }
        ~Bar() { cout << "Bar deconstructor" << endl; }
};

class Base
{
public:
        Base() { cout << "Base constructor" << endl; }
        ~Base() { cout << "Base deconstructor" << endl; }
};

class Derived : public Base
{
public:
        Derived() { cout << "Derived constructor without arguments" << endl; }
        Derived(const Foo &foo, const Bar &bar);
        Derived(const Bar &bar, const Foo &foo);
        ~Derived() { cout << "Derived deconstructor" << endl; }

private:
        Foo m_foo;
        Bar m_bar;
};

Derived::Derived(const Foo &foo, const Bar &bar) :
        m_foo(foo),
        m_bar(bar)
{
        cout << "Derived constructor with argument[Foo foo, Bar bar] passed by references" << endl;
}

Derived::Derived(const Bar &bar, const Foo &foo) :
        m_bar(bar),
        m_foo(foo)
{
        cout << "Derived constructor with argument[Bar bar, Foo foo] passed by references" << endl;
}

int main (int argc, char** argv)
{
        Foo foo;
        Bar bar;

        cout << "test case 1:" << endl;
        Derived deri_1;  //  (1)

        cout << "test case 2:" << endl;
        Derived deri_2(foo, bar);   //  (2)

        cout << "test case 3:" << endl;
        Derived deri_3(bar, foo);   //  (3)

        cout << "test case end" << endl;

        return 0;
}
  

執行結果是:

淺出C++對象模型——理解構造函數、析構函數執行順序


打印出的信息可分為幾部分:

(1)創建對象 foo bar ,執行 Foo,Bar 的構造函數

(2)TestCase1: 創建對象 deri_1,首先執行基類Base 的構造函數,其次執行成員 m_foo,m_bar 的構造函數來構造成員,最后調用自身的構造函數 ( 無參數 )

(3)TestCase2: 創建對象 deri_2,調用順序與case1 順序相同。

(4)TestCase3: 創建對象 deri_3,調用順序與case1 順序相同。注意到 deri_2,deri_3 的創建執行的是不同的 Derived 構造函數,雖然構造函數參數的順序不同,但是構造成員的順序是相同的。

(5)銷毀對象 deri_3,deri_2,deri_1 ,析構函數執行順序相同,與構造對象的順序相反。 C++ 標準規定以對象聲明相反的順序銷毀這些對象。

(6)銷毀對象 bar,foo


編譯運行環境:

    $ uname -a
Linux localhost.localdomain 2.6.18-308.el5 #1 SMP Fri Jan 27 17:17:51 EST 2012 x86_64 x86_64 x86_64 GNU/Linux

$ g++ --version
g++ (GCC) 4.1.2 20080704 (Red Hat 4.1.2-52)
  

References:

[1] 高質量 C++ 編程指南 : http://oss.org.cn/man/develop/c&c++/c/c.htm

[2] http://stackoverflow.com/q/15948381/1145750

轉載本文請注明作者和出處 http://garyelephant.me ,請勿用于任何商業用途!

Author:GaryGao 關注互聯網、自動化、軟件團隊






淺出C++對象模型——理解構造函數、析構函數執行順序


更多文章、技術交流、商務合作、聯系博主

微信掃碼或搜索:z360901061

微信掃一掃加我為好友

QQ號聯系: 360901061

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

【本文對您有幫助就好】

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長會非常 感謝您的哦!!!

發表我的評論
最新評論 總共0條評論
主站蜘蛛池模板: 久久久噜噜噜久久网 | 日操夜操 | 欧美高清在线精品一区 | 日本一级毛片无遮挡 | 全高清特级毛片 | 天天综合射 | 国产农村妇女毛片精品久久 | 欧美成人精品免费播放 | 真人一级一级特黄高清毛片 | 成人一区二区免费中文字幕 | 午夜社区 | 亚洲人xx视频| 国产欧美亚洲精品综合在线 | 4虎在线 | 亚洲精品不卡 | 久久精品国产99国产 | 国产精品视频久久久 | 特级毛片免费视频 | 国产精品视频在线免费观看 | 4虎影院在线观看 | 天天做天天添天天谢 | 久久综合免费 | 国产特级片| 啪啪色视频| 一区二区三区美女视频 | 亚洲欧美日韩在线精品2021 | 国产精品深爱在线 | 久久久久久久久久久观看 | 奇米影视亚洲色图 | 天天草天天草 | 成人夜色视频网站在线观看 | 97在线视频免费观看费观看 | 成人在线视频免费 | 日日噜噜夜夜狠狠视频无 | 日韩欧美在线观看视频一区二区 | 91色综合综合热五月激情 | 国产精品日韩欧美在线第3页 | 2020国产免费久久精品99 | 男人边吃奶边爱边做视频日韩 | 99精品视频在线观看 | 久久嫩模|