C++类与对象中深拷贝与浅拷贝

简介: C++类与对象中深拷贝与浅拷贝

浅拷贝:简单的复制拷贝操作

深拷贝:在堆区重新申请空间,进行拷贝操作

浅拷贝在使用中可能会出现一些问题,如下面这一行代码:

class Person
{
public:
    Person()
    {
        cout << "Person的默认构造函数调用" << endl;
    }
    Person(int age,int height)
    {
        cout << "Person的有参构造函数调用" << endl;
        m_Age = age;
         m_Height = new int(height);
    }
    ~Person()
    {
        //析构代码,将堆区开辟的数据做释放
        if (m_Height != NULL)
        {
            delete m_Height;
            m_Height = NULL;
        }
        cout << "Person的析构构函数调用" << endl;
    }
    int m_Age;       //年龄
    int* m_Height;   //身高
};
void test01()
{
    Person p(18,160);
    cout << "p1的年龄为:" << p.m_Age << " 身高为: "<<*p.m_Height<<endl;
    Person p2(p);
    cout << "p2的年龄为:" << p2.m_Age << " 身高为: " << *p2.m_Height << endl;
}
int main()
{
    test01();
    return 0;
}


这串代码运行时,程序出现了崩溃:

解释:浅拷贝是一个字节一个字节的拷贝,在对身高m_Height进行拷贝时,因为m_Height是一个指针类型的变量,所以是将m_Height的地址拷贝到p2中。在析构函数中,p1,p2都会进行释放,p2先进行释放,此时将m_Height所指向的堆区内存释放。到p1进行释放时,因为m_Height不为空会再次对m_Height进行释放,但由于m_Height已经被p2释放了(堆区内存重复释放),所以会引发错误。


解决:

浅拷贝的问题 要用深拷贝来解决:重新在堆区创建一块内存,将p2的m_Height指向这块新创建的内存

class Person
{
public:
    Person()
    {
        cout << "Person的默认构造函数调用" << endl;
    }
    Person(int age, int height)
    {
        cout << "Person的有参构造函数调用" << endl;
        m_Age = age;
        m_Height = new int(height);
    }
    //自己实现拷贝构造函数,解决浅拷贝带来的问题
    Person(const Person& p)
    {
        cout << "Person拷贝构造函数的调用" << endl;
        m_Age = p.m_Age;
        //m_Height = p.m_Height;       编译器默认实现就是这行代码
        //深拷贝操作
        m_Height = new int(*p.m_Height);
    }
    ~Person()
    {
        //析构代码,将堆区开辟的数据做释放
        if (m_Height != NULL)
        {
            delete m_Height;
            m_Height = NULL;
        }
        cout << "Person的析构构函数调用" << endl;
    }
    int m_Age;       //年龄
    int* m_Height;   //身高
};
void test01()
{
    Person p(18, 160);
    cout << "p1的年龄为:" << p.m_Age << " 身高为: " << *p.m_Height << endl;
    Person p2(p);
    cout << "p2的年龄为:" << p2.m_Age << " 身高为: " << *p2.m_Height << endl;
}
int main()
{
    test01();
    return 0;
}

此时代码就没有出现错误了

目录
相关文章
|
8天前
|
存储 编译器 C++
【c++】类和对象(中)(构造函数、析构函数、拷贝构造、赋值重载)
本文深入探讨了C++类的默认成员函数,包括构造函数、析构函数、拷贝构造函数和赋值重载。构造函数用于对象的初始化,析构函数用于对象销毁时的资源清理,拷贝构造函数用于对象的拷贝,赋值重载用于已存在对象的赋值。文章详细介绍了每个函数的特点、使用方法及注意事项,并提供了代码示例。这些默认成员函数确保了资源的正确管理和对象状态的维护。
35 4
|
9天前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
32 4
|
1月前
|
存储 编译器 对象存储
【C++打怪之路Lv5】-- 类和对象(下)
【C++打怪之路Lv5】-- 类和对象(下)
27 4
|
1月前
|
存储 安全 C++
【C++打怪之路Lv8】-- string类
【C++打怪之路Lv8】-- string类
21 1
|
1月前
|
编译器 C语言 C++
【C++打怪之路Lv4】-- 类和对象(中)
【C++打怪之路Lv4】-- 类和对象(中)
23 4
|
1月前
|
存储 编译器 C++
【C++类和对象(下)】——我与C++的不解之缘(五)
【C++类和对象(下)】——我与C++的不解之缘(五)
|
1月前
|
编译器 C++
【C++类和对象(中)】—— 我与C++的不解之缘(四)
【C++类和对象(中)】—— 我与C++的不解之缘(四)
|
1月前
|
编译器 C语言 C++
C++入门3——类与对象2-2(类的6个默认成员函数)
C++入门3——类与对象2-2(类的6个默认成员函数)
23 3
|
1月前
|
C++
C++番外篇——对于继承中子类与父类对象同时定义其析构顺序的探究
C++番外篇——对于继承中子类与父类对象同时定义其析构顺序的探究
53 1
|
1月前
|
编译器 C语言 C++
C++入门4——类与对象3-1(构造函数的类型转换和友元详解)
C++入门4——类与对象3-1(构造函数的类型转换和友元详解)
19 1