C++中的深拷贝和浅拷贝

时间:2022-04-12 19:48:42
C++中的深拷贝和浅拷贝

当有需要的时候,C++编译器会为类生成默认的内联public复制构造函数,这些复制构造函数只是简单的进行位拷贝,即将一个对象的成员变量的值逐位拷贝给另外一个对象,以完成对象的初始化。
class People
{
public:
    People(int age) 
    {
       this->age = age;
    }
public:
    int age;
};
int main()
{
    int tempAge = 3;
    People ZhangSan(tempAge);
    People LiSi(ZhangSan); 
    return 0;
}
当执行People LiSi(ZhangSan)时就调用了编译器生成的默认复制构造函数,当执行完这句之后LiSi的age变量就和ZhangSan的age的值得一模一样。
这种复制构造函数只是对变量的值进行拷贝。所以 当类中含有指针指针成员时,或者说类中的变量需要动态申请内存时,这种构造复制构造函数就不适用了。
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
class Name
{
public:
    Name(char* name,int length)
    {
        //不能不加this
        this->name = new char[length];
        this->length =length;
        strcpy(this->name,name);
    }
    void printName()
    {
        cout<<"The Address of the Name  "<<(void*)this->name<<endl;
        cout<<this->name<<endl;
    }
    ~Name()
    {
        delete[] this->name;
    }
    void setName(char* name,int length)
    {
        if(this->length <length)
        {
            delete[] name;
            name  = new char[length];
        }
        strcpy(this->name,name);
    }
private:
    char* name;
    int length;
};
int main()
{
    char* temp="ZhangSan";
    Name ZhangSan(temp,strlen(temp));
    Name LiSi(ZhangSan);

    ZhangSan.printName();
    LiSi.printName();

    temp = "Changed";
    ZhangSan.setName(temp,strlen(temp));

    ZhangSan.printName();
    LiSi.printName();
    return 0;
}
如果使用编译器生成的默认的复制构造函数,上述代码的运行结果如下:
The Address of the Name  0x611600
ZhangSan
The Address of the Name  0x611600
ZhangSan
The Address of the Name  0x611600
Changed
The Address of the Name  0x611600
Changed

从上诉结果可以看出:张三的Name跟LiSi的name实质上指向的是同一块内存空间。因为默认复制构造函数的本质就是值拷贝,形式如下:
Name(Name& temp) 
{
    this->name =temp.name;
    this->length = temp.length;
}

在复制的过程中只是单纯的将temp的nam指针复制给当前对象,所以会发生两个对象的name会指向同一块内存地址。
C++中的深拷贝和浅拷贝

ZhangSan 和LiSi的name指向同一块内存区域,当ZhangSan把自己的name修改之后,LiSi的name也会跟着修改,ZhangSan对自己的name的任何修改都会影响LiSi的name。这样是极其危险的,特别容易产生野指针或者释放已经被释放的内存区域。

针对上述这种情况我们自己手动添加复制构造函数如下:
    Name(Name& temp)
    {
        this->name = new char[temp.length];
        this->length = temp.length;
        if(temp.length!=0)
            strcpy(this->name,temp.name);
    }
再次运行上次的程序:
The Address of the Name  0x4c1600
ZhangSan
The Address of the Name  0x4c1630
ZhangSan
The Address of the Name  0x4c1600
Changed
The Address of the Name  0x4c1630
ZhangSan
从结果可以看出ZhangSan的name和LiSI的name指向不同的地址,在执行了setName()之后LiSi的name并没有受到影响。内存分布如下:
C++中的深拷贝和浅拷贝
所以,如果我们创建的类中有指针时,我们就要注意是不是应该给它手动添加复制构造函数,以避免浅拷贝造成个各种问题。