深度复制

      先从一个问题开始探讨:编写一个类,其中有一个动态的字符数组存储字符串,还有一个整型成员存储字符串的长度,并重载<<输出字符串的内容。

      显然,类的定义应该是:

      class Test

{

private:

            char *str;              //字符串的长度是不固定的,需要使用字符指针

            int len;

public:

            Test(const char *s)

            {

                 len = strlen(s);

                 str = new char[len + 1];   //分配足够的内存

                 strcpy(this->str, s);

            }

            Test();

            ~Test()

            {

                 delete [] this->str;

            }

            friend std::ostream &operator<<(std::ostream & os, const Test & st);

};

 

std::ostream & operator<<(std::ostream & os, constTest & st)

{

      os<< st.str;

      returnos;

}

      类的调用例子:

int main()

{

      Test str("helloworld");

      std::cout<<str;

      std::cin.get();

      return 0;

}

      上述程序将字符串常量赋给类的字符串,将自动调用参数为常量字符串的构造函数,在构造函数中分配内存,并将指针指向那块内容,最后在析构的时候释放内容,是没有问题的.

      但是如果调用的是复制构造函数呐?如下代码产生了错误,就是因为没有使用深度复制的原因.

      int main()

{

            Test str1("hello world");

            Test str2 = str1;

            std::cout<<str2;

            std::cin.get();

            return 0;

}

      乍看之下没有问题,将str1指向的内容拷贝到str2指向的内存,两者的数据类型一样的情况下会完美的复制非静态成员.所以std::cout<<str2;会正常调用重载输出str2.str, 而str2.str又会指向str1.str分配的内存,字符串会正常的输出。问题在哪呢?问题在于两个不相关的对象共有同一块内存!

      指针指向的内容是不会存储在类中的,在复制类(调用默认复制函数)的时候,复制的仅仅是指针的值,使两个类指向同一块内容,这将会导致致命的问题.在析构str1时,会释放str1.str指向的内容,这个时候str2.str指向字符串也同时消失,此时任何调用str2.str指向的值将输出不可预期的结果,而在析构str2的时候必定会导致错误,因为不能释放已经释放的内存.

      要想解决这个问题就要放弃浅复制(仅复制指针),使用深复制(复制指针指向的内容),是两个对象完全对立,互不干扰。需要显示定义复制构造函数

      Test(const Test&t)

      {

           this->len =t.len;

           this->str =new char[len + 1];     //指针指向又一块内存

           strcpy(str,t.str);

      }

      这样即使释放了str1.str,str2.str也不会受到影响.

      指针bug绝大部分的问题都来源于把指针误当做实际的内容,应该将指针和其指向的内存一起考虑.

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在Python中,深度复制(deep copy)是指通过copy.deepcopy()方法来实现对象的完全拷贝,包括父对象和子对象。深度复制后的对象与原对象互不相关,对其中一个对象的修改不会影响到另一个对象。深度复制的实现原理是递归地复制对象及其子对象,确保每个对象都是独立的副本。 例如,对于一个包含列表和元组的对象list1,使用copy.deepcopy(list1)进行深度复制后,得到的list2是一个完全独立的副本。对list1进行修改不会影响到list2,同样,对list2的修改也不会影响到list1。 具体实现的代码如下所示: ```python import copy list1 = [[1, 2], (30, 40)] list2 = copy.deepcopy(list1) list1.append(100) list1 = (50, 60) print("list1:", list1) print("list2:", list2) ``` 输出结果为: ``` list1: [[1, 2, 3], (30, 40, 50, 60), 100] list2: [[1, 2], (30, 40)] ``` 可以看到,list1和list2在进行深度复制后,彼此之间是完全独立的。对list1的修改不会影响到list2,反之亦然。 总结起来,深度复制是一种完全拷贝父对象和子对象的方法,确保每个对象都是独立的副本,可以通过copy.deepcopy()方法来实现。<span class="em">1</span><span class="em">2</span><span class="em">3</span><span class="em">4</span> #### 引用[.reference_title] - *1* *4* [python中浅拷贝和深度拷贝教程](https://blog.csdn.net/qdPython/article/details/127752661)[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^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* *3* [Python 深拷贝和浅拷贝详解](https://blog.csdn.net/qq_40630902/article/details/119278072)[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^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值