【C++】深拷贝和浅拷贝 ② ( 默认拷贝构造函数是浅拷贝 | 代码示例 - 浅拷贝造成的问题 )

news/2024/5/19 2:01:48 标签: c++, 拷贝构造函数, 构造函数, 浅拷贝, 深拷贝

文章目录





一、默认构造函数>拷贝构造函数浅拷贝




1、默认构造函数>拷贝构造函数


如果 C++ 类中 没有定义构造函数>拷贝构造函数 , C++ 编译器会自动为该类提供一个 " 默认的构造函数>拷贝构造函数 " , 在函数中对成员变量进行简单的复制操作 ;

" 默认构造函数>拷贝构造函数 " 用于创建一个新对象作为现有对象的副本 , 其作用是将 现有对象 的成员变量 复制到 新对象中 ;

创建一个类对象 并将其 赋值给 另一个类对象时 , 会自动调用 默认构造函数>拷贝构造函数 ;


2、默认构造函数>拷贝构造函数浅拷贝机制


C++ 编译器 为 类 自动生成的 默认构造函数>拷贝构造函数浅拷贝 , 只能拷贝 顶层的 成员变量值 , 如果成员变量 是 引用 或 指针 , 其指向的 类 或 内存空间 中的数据 , 是无法拷贝的 ;


如果 没有定义 构造函数>拷贝构造函数 , 就会触发上述机制 ;

出现如下代码调用时 , 先 调用 有参构造函数 创建了一个 原始对象 s ,

然后 将 s 对象的值 赋值给 s2 对象 , 此时调用的是 构造函数>拷贝构造函数 ,

由于没有定义 构造函数>拷贝构造函数 , 使用的事 C++ 编译器的 默认构造函数>拷贝构造函数 , 进行的拷贝 是 浅拷贝 ;

其中的 字符串指针 , 只拷贝了指针的值 , 没有拷贝字符串的具体内容 ;

	// 调用有参构造函数 , 创建 Student 实例对象
	Student s(18, "Tom");

	// 声明 Student 对象 s2 , 并使用 s 为 s2 赋值
	// 该操作会调用 默认的构造函数>拷贝构造函数 
	// C++ 编译器提供的构造函数>拷贝构造函数 只能进行浅拷贝
	Student s2 = s;




二、代码示例 - 浅拷贝造成的问题



下面代码中 ,

定义的 Student 类 中 , 定义了 有参构造函数 和 析构函数 ,

没有定义构造函数>拷贝构造函数 , 因此 C++ 编译器为其生成了 默认构造函数>拷贝构造函数 ,

默认构造函数>拷贝构造函数浅拷贝 ;


分析下面 创建两个 Student 对象 的代码 :

	// 调用有参构造函数 , 创建 Student 实例对象
	Student s(18, "Tom");
	// 打印 Student 实例对象成员变量值
	s.toString();

	// 声明 Student 对象 s2 , 并使用 s 为 s2 赋值
	// 该操作会调用 默认的构造函数>拷贝构造函数 
	// C++ 编译器提供的构造函数>拷贝构造函数 只能进行浅拷贝
	Student s2 = s;
	s2.toString();

Student s(18, "Tom") 是调用有参参构造函数 , 创建 Student 实例对象 , 并调用 s.toString() 打印上述对象 , 打印结果为 :

m_age = 18 , m_name = Tom

Student s2 = s 代码中 , 声明 Student 对象 s2 , 并使用 s 为 s2 赋值 , 该操作会调用 默认的构造函数>拷贝构造函数 , C++ 编译器提供的构造函数>拷贝构造函数 只能进行浅拷贝 , 因此打印的值是一样的 ;

m_age = 18 , m_name = Tom

分析修改 拷贝对象 代码 :

	// 修改 s2 对象
	strcpy(s2.m_name, "Jey");
	s.toString();
	s2.toString();

strcpy(s2.m_name, "Jey") 代码中 , 修改了 拷贝对象 指针指向的内容 , 将 “Tom” 改为了 “Jey” , 修改了指针指向的内容之后 , 拷贝对象 和 原始对象 的 m_name 成员值都变成了 “Jey” ;


拷贝对象 和 原始对象 都使用了相同的指针 , 那么在析构时就需要注意 , 不能重复 free 掉相同的指针 , 否则就会报错 ;


代码示例 :

#define _CRT_SECURE_NO_WARNINGS

#include "iostream"
using namespace std;

class Student
{
public:

	// 有参构造函数
	Student(int age, const char* name)
	{
		// 获取字符串长度
		int len = strlen(name);

		// 为 m_name 成员分配内存 
		// 注意还要为字符串结尾的 '\0' 字符分配内存
		m_name = (char*)malloc(len + 1);

		// 拷贝字符串
		// C++ 中使用该函数需要
		// 添加 #define _CRT_SECURE_NO_WARNINGS 宏定义
		if (m_name != NULL)
		{
			strcpy(m_name, name);
		}
			
		// 为 m_age 成员设置初始值
		m_age = age;

		cout << "调用有参构造函数" << endl;
	}

	~Student()
	{
		// 销毁 name 指向的堆内存空间
		if (m_name != NULL)
		{
			free(m_name);
			m_name = NULL;
		}
		cout << "调用析构函数" << endl;
	}

	// 该类没有定义构造函数>拷贝构造函数 , C++ 编译器会自动生成默认的构造函数>拷贝构造函数

	// 打印类成员变量
	void toString()
	{
		cout << "m_age = " << m_age << " , m_name = " << m_name << endl;
	}

public:
	int m_age;
	char* m_name;
};

int main()
{
	// 调用有参构造函数 , 创建 Student 实例对象
	Student s(18, "Tom");
	// 打印 Student 实例对象成员变量值
	s.toString();

	// 声明 Student 对象 s2 , 并使用 s 为 s2 赋值
	// 该操作会调用 默认的构造函数>拷贝构造函数 
	// C++ 编译器提供的构造函数>拷贝构造函数 只能进行浅拷贝
	Student s2 = s;
	s2.toString();

	// 修改 s2 对象
	strcpy(s2.m_name, "Jey");
	s.toString();
	s2.toString();

	// 执行时没有问题 , 两个对象都可以正常访问
	// 但是由于拷贝时 执行的是浅拷贝 
	// 浅拷贝 字符串指针时 , 直接将指针进行拷贝 , 没有拷贝具体的值
	// s 和 s2 的 m_name 成员是同一个指针
	// 如果析构时 , 先析构 s2 , 将指针释放了 
	// 之后再析构 s 时 发现 继续释放 被释放的指针 , 报错了



	// 控制台暂停 , 按任意键继续向后执行
	system("pause");
	return 0;
}

执行结果 : 执行后打印如下内容 ,

调用有参构造函数
m_age = 18 , m_name = Tom
m_age = 18 , m_name = Tom
m_age = 18 , m_name = Jey
m_age = 18 , m_name = Jey
请按任意键继续. . .

在这里插入图片描述
按下任意键 , 继续向后执行 , 调用完第一个析构函数后 , 再次尝试调用第二个析构函数 , 报错了 ;

在这里插入图片描述


http://www.niftyadmin.cn/n/5029458.html

相关文章

机器学习——聚类之K-means(未完)

这是我看下来&#xff0c;最简单的内容&#xff0c;哭了&#xff0c;K-means&#xff0c;so nice K-means&#xff0c;由于太过简单&#xff0c;不需要数学推导时&#xff0c;一时间甚至无从下指 首先&#xff0c;K-means需要提前锚定几个点&#xff0c;然后让所有数据样本根据…

Android codec2 编码 -- 基于录屏

文章目录 前言android 原生的应用srcreenrecordMediaCodec获取编码数据流程 前言 本篇文章主要是理解Android 12编码的流程&#xff0c; 首先从上层的应用出发理解mediacodec提供给外部API的用法。然后针对每个api 分析编码各个流程中框架中的流程。 熟悉一个框架的流程 可以…

C#学习 - 操作符

操作符简介 操作符也成为“运算符”&#xff0c;操作符是用来操作数据的&#xff0c;被操作符操作的数据成为操作数&#xff08;Operand&#xff09; 大多数情况下从左向右运算&#xff0c;而赋值与lambda表达式是先运算右边&#xff0c;再运算左边 操作符本质 操作符的本质…

Smart Community(1)之设计规范

通过前面大数据开发相关知识的学习&#xff0c;准备做一个项目进行练习---我给他起了一个响亮的名字&#xff1a;基于HadoopHA的智慧社区服务平台 设计规范&#xff1a; 做一个项目之前肯定要先规定一些开发过程中的设计规范 &#xff08;一&#xff09;数据埋点规范&#xf…

dns电脑服务器发生故障怎么修复

DNS电脑服务器发生故障可能会导致网络连接问题、网页无法访问、或者电子邮件无法发送等情况。修复DNS电脑服务器故障可以采取多种方法&#xff0c;例如检查网络连接、更换DNS服务器等措施。当DNS电脑服务器发生故障时&#xff0c;可以采取以下修复措施&#xff1a; 尝试刷新DNS…

500kV 氧化锌避雷器泄漏电流试验

试验步骤: 1) 记录试验现场的环境温度、 湿度。 2) 对被试设备充分放电后可靠接地。 3) 试验前正确设置安全围栏和悬挂标示牌。 4) 做上节避雷器直流试验, 根据图进行试验接线, 微安表的一端夹子接在上节避雷器的下端, 另一端安装在高压直流发生器的电压输出端; 做中节避雷…

什么是Web组件(Web Components)?它们的主要部分有哪些?

聚沙成塔每天进步一点点 ⭐ 专栏简介⭐ Web 组件&#xff08;Web Components&#xff09;⭐ 自定义元素&#xff08;Custom Elements&#xff09;⭐ 影子 DOM&#xff08;Shadow DOM&#xff09;⭐ HTML 模板&#xff08;HTML Templates&#xff09;⭐ HTML 导入&#xff08;HT…

项目上线部署--》服务器部署流程(一)

目录 &#x1f31f;准备工作 服务器购买 域名购买 域名解析&#xff08;配置 DNS&#xff09; &#x1f31f;服务器环境搭建 配置服务器 安装 CentOS 开发人员相关包 ​编辑 配置免密登陆 &#x1f31f;写在最后 &#x1f31f;准备工作 服务器购买 国内服务器&#x…