研一寒假C++复习笔记--深拷贝和浅拷贝代码实例

文章详细解释了深拷贝和浅拷贝的概念,通过代码实例展示了浅拷贝可能导致的堆区数据重复销毁问题,以及如何通过人为实现深拷贝构造函数来解决这个问题。深拷贝确保了不同对象在堆区拥有独立的内存空间,防止析构时出现错误。
摘要由CSDN通过智能技术生成

目录

1--深拷贝和浅拷贝的基础概念

2--浅拷贝的代码实例

3--深拷贝代码实例

4--参考


1--深拷贝和浅拷贝的基础概念

① 浅拷贝:简单的赋值拷贝操作;

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

2--浅拷贝的代码实例

# include <iostream>

class Person{

public:
    // 有参构造函数
    Person(int age, int height){
        Age = age;
        Height = new int(height); // 存放在堆区
        std::cout << "有参构造函数的调用" << std::endl;
    }

    // 模仿编译器默认实现的浅拷贝构造函数
    Person(const Person &p){
        Age = p.Age;
        Height = p.Height; // 编译器默认实现浅拷贝
        std::cout << "拷贝构造函数的调用" << std::endl;
    }
    
    // 析构函数
    ~Person(){
        if(Height != NULL){ // 手动释放堆区的数据
            delete Height;
            Height = NULL; 
        }       
        std::cout << "析构函数的调用" << std::endl;
    }

    int Age;
    int *Height;
};

int main(){
    Person p1(23, 180);
    std::cout << "p1.Age = " << p1.Age << " p1.Height = " << *p1.Height << std::endl; // 23, 180
    Person p2(p1);
    std::cout << "p2.Age = " << p2.Age << " p1.Height = " << *p2.Height << std::endl; // 23, 180
    return 0;
}

        编译器默认实现的浅拷贝构造函数如下:

    // 模仿编译器默认实现的浅拷贝构造函数
    Person(const Person &p){
        Age = p.Age;
        Height = p.Height; // 编译器默认实现浅拷贝
        std::cout << "拷贝构造函数的调用" << std::endl;
    }

        在创建对象p1时,有参构造函数在堆区申请了内存空间来存储指针 Height 指向的数据;

        在创建对象p2时,使用编译器默认的浅拷贝构造函数,这时p2对象的 Height 也会指向与p1相同的堆区空间;

        但在使用析构函数销毁堆区的数据时,会出现重复销毁的问题,即 p1 和 p2 会重复销毁堆区的同一块内存数据,从而出现 bug 。

3--深拷贝代码实例

        人为定义深拷贝构造函数,可以避免上面浅拷贝带来的堆区数据重复被销毁的问题;

# include <iostream>

class Person{

public:
    // 有参构造函数
    Person(int age, int height){
        Age = age;
        Height = new int(height); // 存放在堆区
        std::cout << "有参构造函数的调用" << std::endl;
    }

    // 人为实现拷贝构造函数,避免浅拷贝
    Person(const Person &p){
        Age = p.Age;
        // Height = p.Height; // 编译器默认实现浅拷贝
        Height = new int (*p.Height);  // 人为实现深拷贝
        std::cout << "拷贝构造函数的调用" << std::endl;
    }
    
    // 析构函数
    ~Person(){
        if(Height != NULL){ // 手动释放堆区的数据
            delete Height;
            Height = NULL; 
        }       
        std::cout << "析构函数的调用" << std::endl;
    }

    int Age;
    int *Height;
};

int main(){
    Person p1(23, 180);
    std::cout << "p1.Age = " << p1.Age << " p1.Height = " << *p1.Height << std::endl; // 23, 180
    Person p2(p1);
    std::cout << "p2.Age = " << p2.Age << " p1.Height = " << *p2.Height << std::endl; // 23, 180
    return 0;
}

        人为实现的深拷贝构造函数如下:

    // 人为实现拷贝构造函数,避免浅拷贝
    Person(const Person &p){
        Age = p.Age;
        // Height = p.Height; // 编译器默认实现浅拷贝
        Height = new int (*p.Height);  // 人为实现深拷贝
        std::cout << "拷贝构造函数的调用" << std::endl;
    }
    

        与编译器默认实现的浅拷贝构造函数不同,深拷贝在堆区重新申请空间来进行拷贝操作,这样不同对象指向的堆区空间不同,在利用析构函数销毁对象时也不会出现重复销毁的问题;

Height = p.Height; // 编译器默认实现浅拷贝
Height = new int (*p.Height);  // 人为实现深拷贝

4--参考

深拷贝和浅拷贝代码讲解

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值