浅析深拷贝和浅拷贝

本文通过实例对比浅拷贝与深拷贝的区别,浅拷贝仅复制数据成员,而深拷贝则为每个对象分配独立内存,避免了多次释放同一内存的问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

要想理解浅拷贝和深拷贝,我们不得不先从它的概念看起;

首先,所谓浅拷贝:就是由默认的拷贝构造函数所实现的数据成员逐一赋值,通常默认的拷贝构造函数式可以胜任次工作的,但若类中含有指针类型的数据,就按数据逐一赋值的发放产生错误。


浅拷贝代码:

#include<iostream.h>
#include<string.h>
class student
{
public:
    student(char *n, float s)

    {

       cout<<"Constucting....<<endl;

       name = new char[strlen(name) +1];

      if(name != NULL)

      {

           strcpy(name, n);
           score = s;
      }
    }
    ~student()

   {

        cout<<"Distructing...<<endl;

       name[0] = '\0';
       delete []name;
   }
private:
    char *name;
    float score;
};


void main()
{
    student stu("LiMing", 90);//调用构造函数,构造对象stu

    student stu1 = stu;      //调用默认的拷贝构造函数,把对象stu的数据成员逐个复制到stu1对应的数据中,但并没有新分配内存空间给stu1;

}


                                                               浅拷贝示意图

主程序结束时,对象逐个撤销,先撤销stu,调用析构函数,用delete释放动态分配的内存空间;撤销对象stu时,第二次调用析构函数,这时因为指针name所致空间已被释放,所以输出随机字符串;当执行delete时,企图释放同一空间,从而导致对同一内存的两次释放,引起错误;

     解决浅拷贝方法:

显示的定义一个自定义的拷贝构造函数,使之不但复制数据成员,而且对对象stu和stu1分配各自的内存空间,这便是深拷贝;


深拷贝代码:

#include<iostream.h>
#include<string.h>
class student
{
public:
    student(char *n, float s)
    {
       name = new char[strlen(name) +1];
      if(name != NULL)
      {
          strcpy(name, n);
          score = s;
      }
   }
    ~student()
    {
       name[0] = '\0';
       delete []name;
    }
    student(student &st)
    {
         cout<<"Copy Conatructing..."<<endl;
         name = new char[strlen(st.name) +1];
         if(name != NULL)
        {
             strcpy(name, st.name);
             score = st.score;
        }
     }
private:
    char *name;
    float score;
};


void main()
{
      student stu("LiMing", 90);
      student stu1 = stu;
}



               深拷贝示意图

       程序开始时,调用构造函数,构造对象stu,用new从内存中动态分配一块空间,字符指针name指向这块空间。
student stu1 = stu;      //调用拷贝构造函数,把对象stu的数据成员逐个复制到stu1对应的数据中,并分配
内存空间给stu1;

   主程序结束时,对象逐个撤销,先撤销stu1,调用析构函数,用delete释放动态分配给stu1的内存空间;撤销对象stu时,第二次调用析构函数,释放了分配给stu的内存空间;当执行delete时,从而避免了对同一内存的两次释放。


深拷贝与浅拷贝不同的是对于引用拷贝的处理,深拷贝将会在新对象中创建和原是对象中对应值类型的字段并且赋值。浅拷贝不会创建新引用类型,会返回相同的类型引用。深拷贝会重新创建新对象,返回新对象的引用字。



来自<<C++面向程序设计>>.

在Python中,可以使用copy模块中的deepcopy方法来实现列表的深拷贝深拷贝会创建一个完全独立的列表副本,无论多少层嵌套,得到的新列表都是原来无关的。可以通过引入copy模块,并使用copy.deepcopy()来进行深拷贝操作。例如: import copy old = [1,[1,2,3],3] new = copy.deepcopy(old) 在上述代码中,old是原始列表,new是深拷贝得到的新列表。无论对new进行任何修改,都不会影响到old的值。这种方法是最安全、最清爽、最有效的深拷贝方法。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [深入浅析Python中list的复制及深拷贝浅拷贝](https://download.youkuaiyun.com/download/weixin_38643269/12867045)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] - *2* *3* [Python中列表List的复制(直接复制、浅拷贝深拷贝)](https://blog.youkuaiyun.com/weixin_49899130/article/details/129380610)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值