C++从零到满绩——类和对象(中)

 

目录

1>>前言

2>>构造函数(我称之为初始化函数)

3>>析构函数(我称之为销毁函数)

4>>拷贝构造函数(我称之为复制函数)

 5>>运算符重载

5.2>>赋值运算符重载

​编辑

6>>结语

1>>前言

        上节课学习了基础篇(包括函数重载、引用、inline、nullptr)and类和对象(上)(包括类的定义、实例化、this指针),忘记的宝子们可以去复习一下~~~

枫の大一-CSDN博客C++从零到满绩——入门基础and类和对象(上)-CSDN博客  跳转上篇

枫の大一-CSDN博客  跳转个人主页查看

        今天我们来类和对象(中)(包括:构造函数、析构函数、拷贝构造函数、赋值运算符重载)。需要注意的是今天我们学习的四个内容 始终 带着两个疑问:

第一:我们不写时,编译器默认生成的函数行为是什么,是否满足我们的需求?

第二:编译器默认生成的函数不满足我们的需求,我们需要自己实现,那么如何自己实现?

ps:如果有觉得小编哪里需要改进的欢迎指出,做出完美的C++篇章肯定少不了各位精神股东的支持啦,谢谢大家~

2>>构造函数(我称之为初始化函数)

        构造函数虽然有构造二字,但它并非开空间造对象,而是在对象实例化时初始化对象。它的本质就是替代之前写的init初始化函数,并且它还能自动调用,简直perfect!

构造函数的特点:

1.函数名域类名一样。

2.无返回值。没有void!!!

3.对象实例化时会自动调用。

4.构造函数可重载。

5.如果类中没有显式定义(我们自己写的)构造函数,编译器会自动生成无参默认构造函数。

6.默认构造包括:无参构造函数、全缺省构造函数、编译器默认生成的构造函数。简单说就是不需要我们传实参就是默认构造。

7.编译器默认生成的构造函数,对于内置类型初始化不确定,看编译器;对于自定义类型成员变量,要求调用这个成员变量的默认构造函数进行初始化。类似于高数的链式求导,y不能对x求导,可以对t求导,t在对x求导。

#include<iostream>
using namespace std;
class Date {
public://公共
	void print() {//打印函数
		cout << _year << "/" << _month << "/" << _day << endl;
	}
private://私有
	int _year;
	int _month;
	int _day;
};
int main() {
	Date d1;
	d1.print();
	return 0;
}

 不写构造函数运行结果为随机值

#include<iostream>
using namespace std;
class Date {
public://公共
	Date(int year = 1, int month = 1, int day = 1) {//构造函数
		_year = year;
		_month = month;
		_day = day;
	}
	void print() {//打印函数
		cout << _year << "/" << _month << "/" << _day << endl;
	}
private://私有
	int _year;
	int _month;
	int _day;
};
int main() {
	Date d1(2024, 11, 24);
	d1.print();
	return 0;
}

写了构造函数对变量d1进行了初始化操作,还用到了上节课的缺省参数 

 

d1这里后面什么都不写

Date d1;

运行结果为1/1/1,因为缺省参数帮助初始化了。 

 

可以发现我们没有进行调用,而是在实质化对象的时候自动调用,这就是它的好用之处。这就是构造函数的使用。

3>>析构函数(我称之为销毁函数)

        析构函数功能与构造函数相反,听名字可以理解为解析构造,将构造函数一步步剖析。它的工作是完成对象中资源的清理释放工作。类似与之前栈使用的Destroy功能。

析构函数的特点:

1.析构函数名是在类名前加上字符~。~本身也是按位取反的意思,这里代表相反。

2.无参数无返回值。不需要void!!!

3.一个类只能有一个析构函数。

4.对象生命周期结束时,系统会自动调用析构函数。

5. 自定义类型的成员变量都会调用析构函数(不管是写的还是默认生成的)。

6.若类无申请资源,析构可以不写,如Date;若类都是自定义成员变量,它们都有自己的析构,也可以不写,如用两个栈实现队列的MyQueue;若有申请资源,一定要写,如用两个栈实现队列的Stack。

7.若有多个对象,后定义的先析构。

#include<iostream>
using namespace std;
typedef int StDataType;
class Stack {
public://公共
	Stack(int n=4) {//构造函数
		_arr = (StDataType*)malloc(sizeof(StDataType) * n);
		if (nullptr == _arr)
		{
			perror("malloc获取失败");
			return;
		}
		_capacity = n;
		_top = 0;
	}
	
	~Stack() {//析构函数
		free(_arr);
		_arr = nullptr;
		_capacity = _top = 0;
	}
private://私有
	StDataType* _arr;
	int _capacity;
	int _top;
};
int main() {
	Stack st;
	return 0;
}

4>>拷贝构造函数(我称之为复制函数)

        拷贝构造函数第一个参数时自身类型的引用,其他参数都是缺省参数。

拷贝构造的特点:

1.拷贝构造函数是构造函数的重载。

2.拷贝构造参数第一个必须是自身类型的引用。

3.自定义类型对象进行拷贝时必须调用拷贝构造,因此传值传参和传值返回的行为都会调用拷贝构造。

4.若未自己写,编译器会自动生成。自动生成的拷贝构造是值拷贝/浅拷贝(一个字节一个字节拷贝),自定义成员变量会调用它的拷贝构造。

5.需要我们写析构的也要写拷贝构造,例如Stack。因为它有申请资源,浅拷贝会导致两块st1和st2都指向同一块空间,因此要进行深拷贝,所以要自己写。

6.传值返回会调用拷贝构造,传值“引用”返回没产生拷贝,因此不需要调用。但是如果返回的引用时一个局部对象,在函数结束时销毁,那么引用会变成野引用,那就不行。传引用返回前提是函数结束后还在。

#include<iostream>
using namespace std;
class Date {
public://公共
	Date(int year = 1, int month = 1, int day = 1) {//构造函数
		_year = year;
		_month = month;
		_day = day;
	}
	void print() {//打印函数
		cout << _year << "/" << _month << "/" << _day << endl;
	}
private://私有
	int _year;
	int _month;
	int _day;
};
int main() {
	Date d1(2024,11,24);
	Date d2(d1);
	d1.print();
	d2.print();
	return 0;
}

 

 用栈的拷贝构造:

//拷贝构造
#include<iostream>
using namespace std;
typedef int StDataType;
class Stack {
public://公共
	Stack(int n = 4) {//构造函数
		_arr = (StDataType*)malloc(sizeof(StDataType) * n);
		if (nullptr == _arr)
		{
			perror("malloc获取失败");
			return;
		}
		_capacity = n;
		_top = 0;
	}

	Stack(const Stack& source) {//拷贝构造函数
		//需要再开辟一个同样大小空间
		_arr = (StDataType*)malloc(sizeof(StDataType) * source._capacity);//乘上之前空间大小
		if (nullptr == _arr)
		{
			perror("malloc获取失败");
			return;
		}
		memcpy(_arr, source._arr, sizeof(StDataType) * source._top);//复制有效数据
		_capacity = source._capacity;
		_top = source._top;
	}

	~Stack() {//析构函数
		free(_arr);
		_arr = nullptr;
		_capacity = _top = 0;
	}
private://私有
	StDataType* _arr;
	int _capacity;
	int _top;
};
int main() {
	Stack st;
	Stack st2(st);
	return 0;
}

 5>>运算符重载

1.这里重载和之前的函数重载区分开,并不是一个意思。

2.当两个类类型对象进行运算符操作时,不能直接操作,需要我们写一个对应的运算符重载函数才可以。

3.它由operator和定义的运算符构成。

4.它的参数个数和运算对象一样多。

5.如果一个重载运算符函数是成员函数,那它少一个参数,因为有隐藏的this指针。

6.特别注意5个运算符不能重载:(.*  ::  sizeof  :?  .)

7.至少有一个类类型参数。

//运算符重载
#include<iostream>
using namespace std;
class Date {
public://公共
	Date(int year = 1, int month = 1, int day = 1) {//构造函数
		_year = year;
		_month = month;
		_day = day;
	}

	bool operator==(const Date& d2) {//重载格式:operator符号
		return _year == d2._year
			&& _month == d2._month
			&& _day == d2._day;
	}
	void print() {//打印函数
		cout << _year << "/" << _month << "/" << _day << endl;
	}
private://私有
	int _year;
	int _month;
	int _day;
};
int main() {
	Date d1(2024,11,24);
	Date d2(d1);
	int r= d1 == d2;
	cout << r;
	return 0;
}

5.2>>赋值运算符重载

        跟拷贝构造区分开,它是两个已经存在的对象进行赋值。

赋值运算符重载特点:

1.必须重载为成员函数,建议带上const。

2.有返回值,写当前类类型的引用,提高效率。

 

//赋值运算符重载
#include<iostream>
using namespace std;
class Date {
public://公共
	Date(int year = 1, int month = 1, int day = 1) {//构造函数
		_year = year;
		_month = month;
		_day = day;
	}

	Date& operator=(const Date& d2) {//重载格式:operator符号
		//跳过自己给自己赋值
		if (this != &d2) {
			_year = d2._year;
			_month = d2._month;
			_day = d2._day;
		}
		//因为d1=d2返回的是d1,而*this也是d1,所以返回*this
		return *this;
	}
	void print() {//打印函数
		cout << _year << "/" << _month << "/" << _day << endl;
	}
private://私有
	int _year;
	int _month;
	int _day;
};
int main() {
	Date d1(2024, 11, 24);
	Date d2(2024, 11, 299);
	d1 = d2;
	d1.print();
	d2.print();
	return 0;
}

6>>结语

        今日份C++入门就先到这里啦,来总结一下:主要讲了类和对象(中)(包括:构造函数、析构函数、拷贝构造函数、赋值运算符重载),感兴趣的宝子们欢迎持续订阅小编,小编在这里谢谢宝子们啦~C++的学习很陡,时而巨难时而巨简单,希望宝子和小编一起坚持下去~你们的三连是我的动力,感谢支持~

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

内网渗透横向移动1

1.信息收集 &#xff08;1&#xff09;判断域控 shell net time /domain shell ping OWA2010CN-God.god.org &#xff08;2&#xff09;主机探测 浏览探测->网络探测 主机列表显示&#xff1a; &#xff08;3&#xff09;域用户收集&#xff1a; shell net user /domain…

Edify 3D: Scalable High-Quality 3D Asset Generation 论文解读

目录 一、概述 二、相关工作 1、三维资产生成 2、多视图下的三维重建 3、纹理和材质生成 三、Edify 3D 1、文本生成多视角图像的扩散模型 2、文本和多视角图像生成法线图像的ControlNet 3、重建与渲染模型 4、多视角高分辨率RGB图像生成 四、训练 1、训练过程 2、…

微软正在测试 Windows 11 对第三方密钥的支持

微软目前正在测试 WebAuthn API 更新&#xff0c;该更新增加了对使用第三方密钥提供商进行 Windows 11 无密码身份验证的支持。 密钥使用生物特征认证&#xff0c;例如指纹和面部识别&#xff0c;提供比传统密码更安全、更方便的替代方案&#xff0c;从而显著降低数据泄露风险…

词云图大师(WordCloudMaster): 探索创意无限的词云世界!

在信息化时代&#xff0c;如何以一种新颖且富有创意的方式表达数据、文字或想法&#xff1f;答案是词云图&#xff01;而词云图大师(WordCloudMaster)&#xff0c;正是您的绝佳选择。 无论是个人创意项目&#xff0c;还是专业工作中的数据可视化&#xff0c;词云图大师都能以强…

pycharm使用debug的时候遇到断点不停的问题

1.首先尝试在程序最开头打断点&#xff0c;检查是否能停下&#xff0c;如果可以&#xff0c;看第二步 2.尝试在你打期望停下的代码附近print("1111111")看看是否输出了这个字符串&#xff0c;验证程序确实走到这一步了 3.如果能走到那一步&#xff0c;但是依然没有…

Epipolar-Free 3D Gaussian Splatting for Generalizable Novel View Synthesis 论文解读

目录 一、概述 二、相关工作 1、单场景3DGS 2、跨场景生成3DGS 3、几何方法解决3D任务 三、eFreeSplat 1、预训练跨视角模块 2、无外极线跨视角交互感知模块 3、迭代跨视角高斯对齐 4、高斯参数预测 一、概述 该论文设计了一种不依赖于极线约束的情况实现可推广的新视…

c++视频图像处理

打开视频或摄像头 打开指定视频 /*VideoCapture(const String &filename, apiPreference);filename:读取的视频或者图像序列的名称apiPreference&#xff1a;读取数据时设置的属性*/ VideoCapture video; //定义一个空的视频对象 video.open("H:/BaiduNetdiskDownlo…

青少年强网杯线上ctf-crypto-wp

目录 AliceAES Classics AliceAES 进入环境&#xff0c;给一个key值和一个iv值 意思是&#xff0c;用这两个值AES编码‘Hello,Bob!’,然后把结果输入进去 把key值和iv值带入解得 然后得出flag Classics 题目是下面这个 根据他解码的顺序&#xff0c;反着写出编码顺序 一开…

工具使用_docker容器_crossbuild

1. 工具简介 2. 工具使用 拉取 multiarch/crossbuild 镜像&#xff1a; docker pull multiarch/crossbuild 创建工作目录和示例代码&#xff1a; mkdir -p ~/crossbuild-test cd ~/crossbuild-test 创建 helloworld.c &#xff1a; #include <stdio.h>int main() …

【Linux系统】—— 基本指令(三)

【Linux系统】—— 基本指令&#xff08;三&#xff09; 1 一切皆文件2 重定向操作2.1 初始重定向2.2 重定向的妙用2.3 追加重定向2.4 输入重定向2.5 一切皆文件与重定向结合 3 Linux 中的文件类型4 日志5 「more」命令6 「less」命令7 「head」与「tail」7.1 查看文件开头和结…

搜索引擎中广泛使用的文档排序算法——BM25(Best Matching 25)

在搜索场景中&#xff0c;BM25能计算每个文档与查询的匹配度&#xff0c;从中找出最相关的文档&#xff0c;并按相关性高低排序展示。 要理解BM25&#xff0c;需要掌握以下几个关键概念&#xff1a; 1. 词频&#xff08;Term Frequency, TF&#xff09;&#xff1a;某关键词在文…

Jupyter Notebook的安装和配置提示功能

Python开发环境搭建conda管理环境-CSDN博客 安装anaconda和对接到编译器的教程可以看上面这一篇 Jupyter Notebook是一种交互式计算环境&#xff0c;它允许用户在单个文档中编写和执行代码、方程、可视化和文本。与其他编译器相比&#xff0c;Jupyter Notebook的突出点在于其交…

Oracle SQL*Plus中的SET VERIFY

在 Oracle SQL*Plus 中&#xff0c;SET VERIFY ON 和 SET VERIFY OFF 是两个用于控制命令执行前后显示变量值的命令。这些命令主要用于调试和验证 SQL 脚本中的变量替换情况。 一、参数说明 1.1 SET VERIFY ON 作用&#xff1a;启用变量替换的验证功能。当启用时&#xff0c;S…

【C】错误的变量定义导致sprintf()‌输出错误

问题描述 刚刚写一个用AT指令透传相关的函数&#xff0c;需要用到sprintf()‌拼接字符串。 结果发现sprintf()‌拼接出来的内容是错误的&#xff0c;简化后的代码如下&#xff1a; const char AT_CIPSEND_FIX_LENGTH_HEADER[11] "ATCIPSEND"; // 错误的&#xff0…

【PHP】部署和发布PHP网站到IIS服务器

欢迎来到《小5讲堂》 这是《PHP》系列文章&#xff0c;每篇文章将以博主理解的角度展开讲解。 温馨提示&#xff1a;博主能力有限&#xff0c;理解水平有限&#xff0c;若有不对之处望指正&#xff01; 目录 前言安装PHP稳定版本线程安全版解压使用 PHP配置配置文件扩展文件路径…

Docker安装RabbitMq详细教程

1.1通过Docker pull RabbitMq docker pull rabbitmq 1.2 获取镜像 docker images 注&#xff1a;执行1.3之前请使用以下命令创建docker网络 docker network create tm 1.3运行命令启动参数 docker run \-e RABBITMQ_DEFAULT_USERrabbitmq \-e RABBITMQ_DEFAULT_PASSrabbitm…

华为ENSP--IP编址及静态路由配置

项目拓扑 项目任务 一、基础配置和IP编址 在AR1、AR2、AR3上配置设备名称和IP地址 # AR1配置 [AR1]interface GigabitEthernet 0/0/0 [AR1-GigabitEthernet0/0/0]ip address 10.0.13.1 24 [AR1-GigabitEthernet0/0/0]q [AR1]interface GigabitEthernet 0/0/1 [AR1-GigabitEth…

老北京香酥芝麻饼

宝安石岩上屋大道有一家老北京香酥芝麻饼&#xff0c;不仅很好吃&#xff0c;还分量特别厚实。应该这家老店&#xff0c;在上屋大道很多人知道和吃过。我每周末都会去买回去给家人一起吃。工作日由于上下班&#xff0c;想买也买不了&#xff0c;因为太晚去老板就收摊了。就像早…

对于相对速度的重新理解 - 2

回到先前说的&#xff0c;先令真空光速为标准光速&#xff0c; 光子的绝对速度 范围&#xff0c; 物质粒子的 范围&#xff0c; 这样的话&#xff0c;我们就可以根据 和 &#xff0c;把速度分成3个段&#xff0c; 这样就可以出现速度和它的负值&#xff0c;也就是速度的矢量具…

GWO-SVMD分解 | Matlab实现GWO-SVMD灰狼算法优化逐次变分模态分解

GWO-SVMD分解 | Matlab实现GWO-SVMD灰狼算法优化逐次变分模态分解 目录 GWO-SVMD分解 | Matlab实现GWO-SVMD灰狼算法优化逐次变分模态分解效果一览基本介绍程序设计参考资料 效果一览 基本介绍 GWO-SVMD灰狼算法优化逐次变分模态分解 内有15种用以优化svmd的适应度函数&#…