当混合默认构造函数和非默认构造函数时,RAII是如何工作的

How does RAII work when mixing default and non-default constructors

本文关键字:构造函数 默认 何工作 工作 RAII 混合      更新时间:2023-10-16

我想围绕使用malloc/free的C库编写一个库包装类(LibWrap)。为此,我想使用C++的RAII来分配和释放内存。我使用lib_address作为一个随机的示例地址,它将从库中接收。然而,当定义我的memeber变量时,以某种方式调用的析构函数具有这个lib_address。

我希望默认构造函数创建的成员变量的析构函数不知道我放入替换成员变量的构造函数中的新地址。

#include <stdlib.h>
#include <iostream>
using namespace std;
class LibWrap
{
    int j;
    int lib_address;
public:
    LibWrap(): //default LibWrap
    j(0),
    lib_address(0)
    {
        cout << "default LibWrap "<<j <<"tt"<<lib_address << "t" << this<<endl;
    }
    LibWrap(int f_j): //special LibWrap
    j(0),
    lib_address(0)
    {
        j = f_j;
        lib_address = rand();
        cout << "special LibWrap " << j<<"t"<< lib_address<< "t" << this <<endl;
    }
    ~LibWrap()
    {
        cout << "killing LibWrap " << j<<"t" <<lib_address <<"t" << this<< endl;
    }
    int g()
    {
        return j;
    }
};
class A
{
    int i;
    LibWrap b;
public:
    A(): //default A
    i(0)
    {
        cout << "default At"<<i << endl;
    }
    A(int f_i)://special A
    i(0)
    {
        i = f_i;
        cout << "special At"<<i << endl;
        b = LibWrap(10);
    }
    ~A()
    {
        cout << "killing At"<<i << endl;
    }

    void p()
    {
        cout <<"Test values: "<< i<< "," << b.g() << endl;
    }
};
int f()
{
    //A a; a.p();
    cout << "variablettlib_addresstreal_address" << endl; 
    A a = A(1);
    cout << "End" << endl;
    //a.p();
}
int main()
{
    f();
}

运行此代码,我希望得到以下结果:

variable        lib_address real_address
default LibWrap 0       0   0xbfef2e28
special A   1
special LibWrap 10  1804289383  0xbfef2df8 
killing LibWrap 10  1804289383  0xbfef2df8 --would expect kiling LibWrap 0  0 0xbfef2e28
End
killing A   1
killing LibWrap 10  1804289383  0xbfef2e28 --would expect killing LibWrap 10 1804289383 0xbfef2df8
b = LibWrap(10);

这不会初始化b。作为构造CCD_ 3的一部分,CCD_。您正在做的是创建一个临时LibWrap,并将该临时复制b中。然后,您将销毁临时LibWrap(这是带有lib_address的额外析构函数调用的来源)。

临时LibWrap是"0xbfef2df8"地址的来源。b变量是"0xbfef2e28"地址。这就是为什么你要把它们按这个顺序排列。