当前位置: 首页 > news >正文

浅拷贝和深拷贝

浅拷贝:

定义:浅拷贝(Shallow Copy)是一种简单的对象复制方式,将一个对象的数据成员直接复制给另一个对象(通常是通过默认的复制构造函数或赋值运算符实现),这些数据成员可以是基本类型或指向其他对象或资源的指针。具体地说,浅拷贝只是复制指针的值,而不是复制所指向的内存区域。

因此,如果被浅拷贝的对象存在一个或多个指针成员,则会导致多个对象共享同一块内存区域。当其中任何一个对象析构时,该指针将被 delete 解除绑定的内存区域,从而导致其他引用的该内存区域变得无效,从而产生不可预期的行为(如悬挂指针、内存泄漏等)。

=》缺点总结:

1.多次析构(浅拷贝n次就析构n+1次)

解决方法:增加一个引用计数,每个对象析构时,--引用计数,最后一个析构的对象释放空间

2.一个对象修改影响另一个对象

解决方法:写时拷贝(本质,延迟拷贝:谁去写,谁做深拷贝)

没人写不就赚了么

浅拷贝示例:

#pragma once
#include <iostream>
#include <assert.h>
using namespace std;class mystring
{
public://全缺省构造函数mystring(const char* str = "")//""是常量字符串,常量字符串默认自带\0//受声明顺序的影响/* :_size(strlen(str)),_capacity(_size),_str(new char[_capacity+1])*/{//不受声明顺序的影响_size = strlen(str);_capacity = _size;_str = new char[_capacity + 1];strcpy(_str, str);}//析构函数~mystring(){delete[] _str;_str = nullptr;_size = _capacity = 0;}private:size_t _size;size_t _capacity;char* _str;
};//拷贝构造
int main()
{mystring s1("hello world");mystring s2(s1);return 0;
}

文字说明: 

s2析构结束后,s1析构s2刚销毁的空间。故会发生警告

深拷贝:

定义:深拷贝(Deep Copy)是一种对象复制方式,将一个对象或资源以及其指向的某些对象或资源一起复制给另一个对象。具体地说,深拷贝会为新对象重新分配内存空间,并将被复制对象的数据成员递归地复制到新对象中。

因此,如果被深拷贝的对象存在指针等非基本类型的成员变量,那么深拷贝会额外申请一块与原对象所使用的内存区域大小相同的内存,将其中的数据递归地复制到新对象中,并返回指向该内存区域的指针。

示例:

#pragma once
#include <iostream>
#include <assert.h>
using namespace std;class mystring
{
public:// 全缺省构造函数mystring(const char* str = ""): _size(strlen(str)), _capacity(_size), _str(new char[_capacity + 1]){strcpy(_str, str);}// 拷贝构造函数mystring(const mystring& str): _size(str._size), _capacity(str._capacity), _str(new char[_capacity + 1]){strcpy(_str, str._str);}// 析构函数~mystring(){delete[] _str;_str = nullptr;_size = _capacity = 0;}private:size_t _size;size_t _capacity;char* _str;
};int main()
{mystring s1("hello world");mystring s2(s1);return 0;
}

图解:

文字说明:

在拷贝构造函数中,我们使用深拷贝方式将 _str 字符指针所指向的内存区域递归地复制到新对象中,并确保在新对象销毁时释放所申请的内存空间。这样,当新对象与原对象均被销毁时,它们所使用的内存空间是相互独立的,避免了内存泄漏和悬挂指针等问题

http://www.lryc.cn/news/64815.html

相关文章:

  • 进程地址空间与页表方面知识点(缺页中断及写时拷贝部分原理)
  • Photoshop如何使用滤镜之实例演示?
  • Flutter 组件抽取:日期(DatePicker)、时间(TimePicker)弹窗选择器【仿照】
  • 基于opencv的YOLOV3对图片的目标检测
  • Mermaid流程图
  • 国产!全志科技T507-H工业核心板( 4核ARM Cortex-A5)规格书
  • java小记 2023-05-05
  • CentOS安装Nginx
  • CSS布局基础(CSS书写顺序 导航栏写法 常见问题)
  • 打造卓越 QML 层级设计:从入门到精通
  • shell流程控制之条件判断练习
  • linux中TF启动卡制作:磁盘分区文件同步
  • 【操作系统OS】学习笔记:第一章 操作系统基础【哈工大李治军老师】
  • Linux C/C++ 网络编程中地址格式转换(inet_pton和inet_ntop函数)
  • 庖丁解牛函数知识---C语言《2》
  • Git 使用教程:最详细、最正宗手把手教学(万字长文)
  • 【华为OD机试 2023最新 】最优资源分配/芯片资源占用(C语言题解 100%)
  • markdown二元运算符
  • 【华为/华三】PPP
  • 【Java笔试强训 9】
  • 【C++】STL标准库之list
  • Nomogram | 盘点一下绘制列线图的几个R包!~(二)
  • Django之定时任务django-crontab
  • linux常见命令
  • 【14.HTML-移动端适配】
  • 平衡二叉树旋转机制
  • 深入浅出C++ ——C++11
  • 智能座舱3.0阶段,看全球巨头如何打造更具“价值”的第三空间
  • 【Linux】入门介绍
  • 【Python】序列类型②-元组