【C++】深拷贝和浅拷贝

news2025/1/11 0:47:21

目录

浅拷贝

 深拷贝

字符串的构造

例有两个类的深拷贝:


浅拷贝

在类中,若我们不写拷贝构造函数,则程序会提供一个默认的拷贝构造函数,该函数为浅拷贝

   //默认拷贝构造 ---浅拷贝--值复制

类名(const 类名& 形参名)   {成员1= 形参名.成员1; }

class A
{
public:
	A(int i = 0, int j = 0) :m_i(i), m_j(j) {}//构造函数
	void print() { cout << m_i << "," << m_j << endl; }
private:
	int m_i;
	int m_j;
};
void main()
{
	A a(4, 7);
	A b(a);//使用默认拷贝构造函数
	a.print();//4,7
	b.print();//4,7
}

 

浅拷贝将一个对象的所有信息直接拷贝到另一个对象中,如果是值则拷贝值;如果是引用、指针则拷贝其内存地址,这时,两个对象的该成员共用一个地址。

由于对象最后都需要析构,若浅拷贝的对象中有指针,该地址在析构的时候将会进行两次(拷贝和被拷贝的对象每个一次),导致程序出错。

class A
{
public:
	A(int i = 0)
	{
		//m_i = new int(i);
		m_i = new int;
		*m_i = i;
	}
	~A()
	{
		if (m_i != NULL)
		{
			delete m_i;
			m_i = NULL;
		}

	}
	A(const A& a)  //默认拷贝构造 ---浅拷贝--值复制
	{
		m_i = a.m_i;  //两个对象的指针指向同一块内存单元
	}
	void print()
	{
		cout << *m_i << endl;
	}
private:
	int* m_i;
};
void main()
{
	A a(10);
	a.print();
	A b(a);  //调用默认拷贝构造,会导致b对象中的m_i和a对象中的m_i指向同一块内存空间
	b.print();
	//在即将退出时,调用析构会出问题,析构了两次
}

 深拷贝

如果有指针作为数据成员,则必须要写析构函数拷贝构造函数,拷贝资源。

深拷贝中如果对象有申请动态内存,则深拷贝中也会新申请一个(属性拷贝),析构函数中释放空间。

class A
{
public:
	A(int i = 0)
	{
		m_i = new int;
		*m_i = i;
	}
	A(const A& a)  //深拷贝
	{
		m_i = new int;//开新空间
		*m_i = *(a.m_i);
	}
	~A()
	{
		delete m_i;
		m_i = NULL;
	}
	void print() { cout << *m_i << endl; }
private:
	int* m_i;
};
void main()
{
	A a(20);
	A b(a);
	a.print();//20
	b.print();//20
}

字符串的构造

strcpy拷贝所有字符串
strncpy拷贝n个字符串
memcpy  void*memcpy(void*dest,void*src,count)都可以拷贝

class STR
{
public:
	STR(const char* str)
	{
		m_str = new char[strlen(str) + 1];
		strcpy_s(m_str, strlen(str) + 1, str);
	}
	STR()//什么都不传入,拿一个\0占位
	{
		m_str = new char[1];
		m_str[0] = '\0';
	}
	STR(int n, char c)
	{
		m_str = new char[n + 1];
		memset(m_str, c, n);//strcpy只能拷贝字符串,memset什么都可以拷贝
		m_str[n] = '\0';
	}
	STR(const STR& s)//深拷贝
	{
		m_str = new char[strlen(s.m_str) + 1];
		strcpy_s(m_str, strlen(s.m_str) + 1, s.m_str);
	}
	~STR()//析构
	{
		if (m_str != NULL)
		{
			delete[]m_str;
			m_str = NULL;
		}
	}
	void print()
	{
		cout << m_str << endl;
	}
private:
	char* m_str;
};
void main()
{
	STR s1("12345");
	STR s2;
	STR s3(4, 'k');
	STR s4(s1);
	s1.print();
	s2.print();
	s3.print();
	s4.print();
}

如果成员是指针可以new大小(上方代码),但数组不能,只能依靠数组定大小截断(如下)

就是说指针需要先申请空间(申请需要放入的大小)再往里面放东西,而数组因为本身有大小所以直接放入数组能存的大小的内容。

class STR
{
public:
	STR(const char* str)
	{
		int n = sizeof(m_str);
		memcpy(m_str, str, n - 1);
		m_str[n - 1] = '\0';
	}
	void print()
	{
		cout << m_str << endl;
	}
private:
	char m_str[20];
};
void main()
{
	STR a("asdf");
	STR b("asdfasdfasdfasddfasdfasdfadfgsdfgsdfgsdfg");
	a.print();
	b.print();
}

例有两个类的深拷贝:

class Date
{
private:
	int m_year;
	int m_month;
	int m_day;
public:
	Date(int y, int m, int d) :m_year(y), m_month(y), m_day(d) {}
	void show()
	{
		cout << m_year << " /" << m_month << "/" << m_day << endl;
	}
	Date(const Date& d) :m_year(d.m_year), m_month(d.m_month), m_day(d.m_day)
	{
		cout << "Data(&)" << endl;
	}
};
class Student
{
private:
	int m_num;
	char* m_name;
	char m_sex;
	Date m_birthday;
public:
	Student(int num, const char* name, char sex, int y, int m, int d) :m_num(num), m_sex(sex), m_birthday(y, m, d)//没有默认值必须显示的调用构造函数
	{
		m_name = new char[strlen(name) + 1];
		strcpy_s(m_name, strlen(name) + 1, name);
	}
	void print()
	{
		cout << m_num << " " << m_name << " " << m_sex;
		m_birthday.show();
	}
	~Student()
	{
		if (m_name != NULL)
		{
			delete[]m_name;
			m_name = NULL;
		}
	}
	Student(const Student& s) :m_num(s.m_num), m_sex(s.m_sex), m_birthday(s.m_birthday) //调用Date类的拷贝构造
	{
		m_name = new char[strlen(s.m_name) + 1];
		strcpy_s(m_name, strlen(s.m_name) + 1, s.m_name);
	}
};

void main()
{
	Student s1(1001, "wangpangpang", 'm', 2000, 12, 24);
	Student s2(s1);
	s2.print();
	s1.print();
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/112882.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【实时数仓】DWS层之商品主题计算、地区主题表(FlinkSQL)

文章目录一 DWS层-商品主题计算1 把JSON字符串数据流转换为统一数据对象的数据流&#xff08;1&#xff09;转换订单宽表流数据&#xff08;2&#xff09;转换支付宽表流数据2 把统一的数据结构流合并为一个流&#xff08;1&#xff09;代码&#xff08;2&#xff09;测试3 设定…

216. 组合总和 III

216. 组合总和 III 找出所有相加之和为 n 的 k 个数的组合&#xff0c;且满足下列条件&#xff1a; 只使用数字1到9每个数字 最多使用一次 返回 所有可能的有效组合的列表 。该列表不能包含相同的组合两次&#xff0c;组合可以以任何顺序返回。 示例 1: 输入: k 3, n 7 输…

3.2 Static Terrestrial Laser Scanners 静态地基激光扫描仪

本章节介绍的静态地基激光扫描系统指的是那些在一个固定位置的位置上对周边场景地物特征进行扫描的设备。该类型设备的扫描测量机制是&#xff0c;通过激光测距仪进行斜距测量&#xff0c;与此同时通过水平和竖直两个方向上同步运动的角度编码器来记录角度变化值&#xff08;如…

编译原理——正规式、NFA构造DFA

一、DFA和NFA的区别 NFA&#xff1a;非确定有限自动机 DFA&#xff1a;确定有限自动机 NFA在同一状态&#xff0c;可以有多条出边&#xff0c;DFA在同一状态&#xff0c;只能有一条出边&#xff1b; NFA的初态可以具有多个&#xff0c;DFA的初态是唯一的&#xff1b; 比如这个…

数据结构入门——栈和队列详解

栈和队列详解1 栈1.1 栈的概念及结构1.2 栈的实现1.3 支持动态增长的栈1.3.1 结构声明1.3.2 栈的初始化和销毁1.3.3 入栈和出栈操作1.3.4 栈的判空和元素个数2 队列2.1 队列的概念及结构2.2 队列的实现2.3 链表实现队列2.3.1 结构声明2.3.2 队列的初始化和销毁2.3.3 队列入队和…

【工作流Activiti7】4、Activiti7 结束/终止流程

1. 结束/终止 正在运行的流程实例 思路&#xff1a;跟回退一样的思路一样&#xff0c;直接从当前节点跳到结束节点&#xff08;EndEvent&#xff09; /*** 结束任务* param taskId 当前任务ID*/ public void endTask(String taskId) {// 当前任务Task task taskService…

Tomcat学习

文章目录1、Tomcat是什么&#xff1f;2、Tomcat安装部署java环境部署tomcat目录结构介绍webapps目录bin目录tomcat启停方式3、Tomcat配置文件tomcat-users.xmlserver.xml结构组件详情配置文件注释4、Tomcat端口5、JVM调优6、Tomcat启动慢解决7、Tomcat面试题目1.Tomcat的默认端…

npm : 无法将“npm”项识别为 cmdlet、函数、脚本文件或可运行程序的名称

一、问题描述 首次用vscode运行vue项目时&#xff0c;报错&#xff1a; npm : 无法将“npm”项识别为 cmdlet、函数、脚本文件或可运行程序的名称。 二、解决 我的解决过程&#xff1a;检查是否安装node.js环境 已安装node 这样一来&#xff0c;真不知道怎么回事了。环境也没有…

五、Arduino IDE开发esp8266环境搭建

1、安装驱动程序 (1)安装USB转串口驱动程序。 (2)根据板载的USB转串口驱动芯片选择合适驱动安装。USB转串口芯片负责和电脑之间进行数据通信。 (3)常见USB转串口驱动 CP210x驱动:CP210x USB 至 UART 桥 VCP 驱动器 - 芯科科技 CH340驱动 2、Arduino IDE环境搭建 要想使用Ar…

数据结构与算法java实战篇--高级排序

目录 一.希尔排序 二.划分 三.快速排序 1. 快速排序的算法 2.选择枢纽 一.希尔排序 希尔排序是基于插入排序的算法来实现的&#xff0c;不同的是希尔排序是采用n-增量来实现排序&#xff0c;如下是希尔排序的图解&#xff1a; 希尔排序会先以n个增量对元素进行划分&#xf…

原神私服 grasscutter搭建及食用教程 v3.3

本教程搭建过程食用vmware虚拟机服务端搭建过程及其简单。照着教程操作即可。本次对应的版本是3.3的版本&#xff0c;后期会持续更新。 一.资源下载准备&#xff1a; 1.vmwera16虚拟机下载安装自己百度吧&#xff0c;非常简单。一路next安装完后再输入一个百度来的秘钥即可。…

【kafka】学习笔记(二)

学习笔记五、Kafka Broker5.1、在zookeeper的服务端存储的Kafka相关信息5.2、Kafka Broker 总体工作流程5.3、Kafka Broker 节点服役和退役5.3.1、节点服役5.3.2、节点退役5.4、Kafka Broker 副本5.4.1、副本信息5.4.3、Leader 选举流程5.4.3、 Leader 和 Follower 故障处理细节…

【OpenCV-Python】教程:7-6 SVM识别手写字符

OpenCV Python SVM 识别手写字符 【目标】 用 SVM 识别手写字符 【代码】 在kNN中&#xff0c;直接用的是像素亮度值&#xff0c;这次&#xff0c;我们将使用 Histogram of Oriented Gradients (HOG) 作为特征向量 import cv2 import numpy as npSZ 20 bin_n 16 # Numbe…

Python pandas有几千个库函数,你用过几个?(5)

上一篇链接&#xff1a; Python pandas有几千个库函数&#xff0c;你用过几个&#xff1f;&#xff08;4&#xff09;_Hann Yang的博客-CSDN博客 12个pandas子模块又包含310个库函数&#xff08;含类、方法、子模块&#xff09;&#xff1a; import pandas as pd funcs [_ …

【C与数据结构】——寒假提高每日练习Day1

一共16日的练习&#xff0c;分为选择题与编程题&#xff0c;涵盖了C语言所学以及数据结构的重点&#xff0c;以及一些秋招、春招面试的高频考点&#xff0c;难度会随着天数而上升。 &#xff08;建议在电脑客户端进行&#xff0c;将鼠标选中被遮挡的地方&#xff0c;即可看到解…

aws codepipeline创建跨账户的cicd

参考资料 Building a Cross-account CI/CD Pipeline Create a pipeline in CodePipeline that uses resources from another AWS account 通常来说&#xff0c;我们会将代码和pipeline配置不同的账户中&#xff0c;在codepipeline的source阶段指定为另一个账号的codecommit仓…

将扩散模型应用到文本领域

前言 扩散模型在文生图领域可谓是大显身手&#xff0c;效果棒棒&#xff5e; 每当一个idea有效之时&#xff0c;便会有更多相关的研究跟进尝试&#xff0c;今天就给大家介绍一篇将扩散模型应用到文本生成领域的工作&#xff0c;这也是一篇比较新的paper&#xff0c;其中还用到了…

LCS+LIS最长公共上升子序列

LIS LCS AcWing 272. 最长公共上升子序列 没优化的代码 优化解释在代码注释中优化解释在代码注释中优化解释在代码注释中 #include<iostream> #include<cstring> #include<algorithm>using namespace std;const int N 3e3 10;int a[N], b[N], f[N][N];i…

开启浏览器sourcemap调试生产环境代码

开启浏览器sourcemap调试生产环境代码 Source Map介绍 在做网站性能优化的时候&#xff0c;我们经常会做js和css代码压缩。但是压缩之 后的代码在调试的时候就会异常困难。source map就是解决问题的一种解决方案 浏览器Source Map 浏览器可以设置开启或者关闭SourceMap&…

78. 子集

78. 子集 给你一个整数数组 nums &#xff0c;数组中的元素 互不相同 。返回该数组所有可能的子集&#xff08;幂集&#xff09;。 解集 不能 包含重复的子集。你可以按 任意顺序 返回解集。 示例 1&#xff1a; 输入&#xff1a;nums [1,2,3] 输出&#xff1a;[[],[1],[2]…