2022年

2022年发布的文章
  • C++构造函数、析构函数和变量的生存期

    构造函数在对象生成时会被调用,析构函数在对象消亡时会被调用。对象何时生成和消亡是由对象的生存期决定的。下面通过一个例子来加深对构造函数、析构函数和变量的生存期的理解。

    #include <iostream >
    using namespace std;
    class Demo {
        int id;
    public:
        Demo(int i)
        {
            id = i;
            cout << "id=" << id << "constructed" << endl;
        }
        ~Demo()
        {
            cout << "id=" << id << "destructed" << endl;
        }
    };
    Demo d1(1);
    void Func()
    {
        static Demo d2(2);
        Demo d3(3);
        cout << "func" << endl;
    }
    int main()
    {
        Demo d4(4);
        d4 = 6;
        cout << "main" << endl;
        {
            Demo d5(5);
        }
        Func();
        cout << "main ends" << endl;
        return 0;
    }

    运行结果(行号只是为了便于查看,它不是输出的一部分):
    01) id=1constructed
    02) id=4constructed
    03) id=6constructed
    04) id=6destructed
    05) main
    06) id=5constructed
    07) id=5destructed
    08) id=2constructed
    09) id=3constructed
    10) func
    11) id=3destructed
    12) main ends
    13) id=6destructed
    14) id=2destructed
    15) id=1destructed

    要分析程序的输出,首先要看有没有全局对象。因为全局对象是进入 main 函数以前就形成的,所以全局对象在 main 函数开始执行前就会被初始化。

    本程序第 16 行定义了全局对象 d1,因此 d1 初始化引发的构造函数调用,导致了第 1) 行的输出结果。

    main 函数开始执行后,局部对象 d4 初始化,导致第 2) 行输出。

    第 26 行,d4=6;,6 先被自动转换成一个临时对象。这个临时对象的初始化导致第 3) 行输出。临时对象的值被赋给 d4 后,这条语句执行完毕,临时对象消亡,因此引发析构函数调用,导致第 4) 行输出。

    第 29 行的 d5 初始化导致第 6) 行输出。d5 的作用域和生存期都只到离它最近的,且将其包含在内的那一对{}中的}为止,即第 30 行的},因此程序执行到第 30 行时 d5 消亡,引发析构函数调用,输出第 7) 行。

    第 8) 行的输出是由于进入 Func 函数后,执行第 19 行的静态局部对象 d2 初始化导致的。

    静态局部对象在函数第一次被调用并执行到定义它的语句时初始化,生存期一直持续到整个程序结束,所以即便 Func 函数调用结束,d2 也不会消亡。

    Func 函数中的 d3 初始化导致了第 9) 行输出。

    第 31 行,Func 函数调用结朿后,d3 消亡导致第 11) 行输出。

    main 函数结束时,其局部变量 d4 消亡,导致第 13) 行输出。

    整个程序结束时,全局对象 d1 和静态局部对象 d2 消亡,导致最后两行输出。

更多...

加载中...