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

CPP从入门到入土之类和对象Ⅲ

拷贝构造函数

拷贝构造函数是一个已经存在的对象初始化一个新的对象时,调用的函数
例如:
假设我有一个盒子,里面装了一个苹果

拷贝构造函数的特点

  • 拷贝构造函数是构造函数的一个重载
  • 拷贝构造函数的第一个参数必须是类类型对象的引用,例如一个日期类的拷贝构造:
// err: “Date”: 非法的复制构造函数: 第一个参数不应是“Date”
// Date(Date d);

// 拷贝构造函数
Date(Date& d) {
	_year = d._year;
	_month = d._month;
	_day = d._day;
}
  • 如果使用传值调用的方式,编译器会直接报错,因为会引发无穷递归
    在这里插入图片描述

拷贝构造函数也可以有多个参数,但是第一个参数必须是类类型对象的引用,后面的参数必须要有缺省值

  • cpp规定自定义类型对象进行拷贝行为必须调用拷贝构造,所以上图的自定义类型传值传参和传值返回都会调用拷贝构造
  • 如果没有显式定义拷贝构造,编译器会默认生成拷贝构造函数 自动生成的拷贝构造对内置类型成员变量会完成浅拷贝(又叫值拷贝) 即一个字节一个字节拷贝,对自定义类型成员变量会调用它的拷贝构造

我们需要详细讲一下浅拷贝和深拷贝

浅拷贝与深拷贝

  1. 默认拷贝构造函数的行为
// 浅拷贝和深拷贝
class Stack {
public:
	// 没有显式写拷贝构造,默认生成的拷贝构造:
	Stack(Stack& ST) {
		_a = ST._a; // 直接复制指针地址,浅拷贝
		_size = ST._size;
		_capacity = ST._capacity;
	}
private:
	int* _a;
	int _size;
	int _capacity;
};

// 假设有以下代码,其实编译器已经报错了
int main() {
	Stack st1;
	st1.Push(1);
	st1.Push(2);

	Stack st2 = st1;  // 调用默认拷贝构造函数(浅拷贝)
}
  1. 问题分析
    这是报错信息:
    ![[QQ_1742990601993.png]]

我们不管报错,来分析一下
此时,st1st2的成员变量状态如下:

  • st1._ast2._a指向同一块内存地址(浅拷贝直接复制了指针值)
  • st1._sizest2._size 相同。
  • st1._capacityst2._capacity 相同
  1. 析构时崩溃的原因
    st1st2 离开作用域时,它们的析构函数会被依次调用:
~Stack() {
    free(_a);  // 释放 _a 指向的数组
    _a = nullptr;
}

具体过程:

  • st2 先析构
    st2._a 指向的内存被释放。

  • st1 再析构
    st1._a 现在指向一块已经被释放的内存,再次调用 free() 会导致 重复释放(double free),引发程序崩溃

  1. 图解
    在这里插入图片描述

  2. ** 解决方案:深拷贝**
    Stack 类显式实现深拷贝构造函数,让每个对象拥有独立的资源:

	//1. 分配内存
		_a = (int*)malloc(sizeof(ST.capacity));
		 if(_a == nullptr){
			 perror("malloc() err");
			 return;
		 }	
	 // 2. 复制数据
		 memcpy(_a, ST._a, sizeof(int) * ST._size);
	 // 3. 复制其他成员
		 _size = ST._size;
		 _capacity = ST._capacity;
}

在编译器生成的默认拷贝构造函数中,内置类型是按照字节方式直接拷贝的,而自定义类型是调用其拷贝构造函数完成的
6. 总结

  • 如果没有需要管理的资源,一般情况下不写拷贝构造函数默认生成的即可,例如日期类
  • 如果都是自定义类型成员,内置类型成员没有指向资源,默认生成的即可
  • 一般情况下,不用显式写析构函数,就不用写拷贝构造
  • 如果内部有指针或者一些值指向的资源,需要显式写析构释放,就要显式写构造完成深拷贝

野引用

传值返回会产生一个临时对象调用拷贝构造,传引用返回,返回的时返回对象的别名(引用),没有产生拷贝。但是如果返回对象是一个当前函数局部域的局部对象,函数结束之后就销毁了,那么使用引用返回就是有问题的,这时的引用相当于一个野引用,类似野指针。传引用返回会减少拷贝,但是一定要确保返回对象在函数结束之前还存在,才能传引用返回

// 传值返回
Date Func1() {
	Date tmp(2024, 7, 5);
	tmp.Print();
	return tmp;
}
// 传引用返回
Date& Func2(){
	Date tmp(2024, 7, 5);
	tmp.Print();
	return tmp;
}

int main(){
	// Func返回了一个局部对象tmp的引用作为返回值
	// Func2函数结束,tmp对象被销毁,相当于野引用
	Date ret2 = Func2();
	ret2.Print();
	cout << "*******************************" << endl;
	Date ret1 = Func1();
	ret1.Print();

	return 0;
}

输出结果:在这里插入图片描述

运算符重载

未完待续~~

相关文章:

  • UMI-OCR Docker 部署
  • Python:计算机二级:简单应用
  • g对象在flask中主要是用来实现什么
  • 【Linux】Linux_Ubuntu与Windows之间的文件传输
  • 3.26品优购
  • Linux之编辑器vim命令
  • 力扣HOT100之普通数组:53. 最大子数组和
  • Linux编译器gcc/g++使用完全指南:从编译原理到动静态链接
  • 【leetcode hot 100 215】数组中的第K个最大元素
  • kubeadm部署k8s-1.32版本集群(1个master,1个worker)
  • PX4飞控-接收MAVLINK消息(2)-生成MAVLINK_MSG_ID_***.h文件
  • QEMU源码全解析 —— 块设备虚拟化(10)
  • [笔记] 系统分析师 第二章 经济管理与应用数学 (未完待续)
  • Linux系统离线安装ollama【详细版】
  • <command-line>:0:1: error: macro names must be identifiers m
  • 2000-2019年各省地方财政行政事业性收费收入数据
  • 【数据采集】技术对比:PCIe、PXIe、PCI、PXI、网口与USB
  • 线上分享会 如何用deepseek和豆包等AI平台获客?
  • 基于SSM+Vue物流信息管理系统(附源码)
  • 信竞资讯
  • 李在明正式登记参选下届韩国总统
  • 习近平会见委内瑞拉总统马杜罗
  • 玉渊谭天丨一艘航母看中国稀土出口管制为何有效
  • 七大交响乐团“神仙斗法”,时代交响在上海奏出时代新声
  • 牛市早报|国家发改委:今年将推出约3万亿元优质项目,支持民营企业参与
  • 马上评|比余华与史铁生的友情更动人的是什么