C++之类和对象的中篇

𝙉𝙞𝙘𝙚!!👏🏻‧✧̣̥̇‧✦👏🏻‧✧̣̥̇‧✦ 👏🏻‧✧̣̥̇:Solitary_walk

      ⸝⋆   ━━━┓
     - 个性标签 - :来于“云”的“羽球人”。 Talk is cheap. Show me the code
┗━━━━━━━  ➴ ⷯ

本人座右铭 :   欲达高峰,必忍其痛;欲戴王冠,必承其重。

👑💎💎👑💎💎👑 
💎💎💎自💎💎💎
💎💎💎信💎💎💎
👑💎💎 💎💎👑    希望在看完我的此篇博客后可以对你有帮助哟

👑👑💎💎💎👑👑   此外,希望各位大佬们在看完后,可以互相支持,蟹蟹!
👑👑👑💎👑👑👑 

目录:
一:类的6个默认成员函数
二:构造函数
三:析构函数
四:拷贝构造函数
五:赋值运算符重载
六:const成员函数
七:取地址以及const 取地址操作符重载

思维导图:

1:类的6个默认成员函数

1)空类:一个类里面什么成员也没有,既称之为空类

2)一个空类里面并不是什么也没有,编译器会默认生成6个默认成员函数(不需要用户自己定义,编译器隐式调用的)

3)

6个默认成员函数

构造函数:主要就是完成函数的初始化

析构函数:对资源的清理

拷贝构造函数:使用同类对象初始化并创建同类对象

赋值运算符重载:把一个对象赋值给另一个对象

const成员函数:

取地址以及const 取地址操作符重载:普通对象和const 对应的对象进行取地址

2:构造函数

对于C++而言,我们可以不用 手动调用初始化函数,构造函数就可以完成这一工作

在创建对象 的同时编译器会默认自动调用构造函数

关于构造函数基本使用方法:

1)构造函数 的名字和类的名字相同

2)构造函数没有返回类型(注意没有返回类型不代表就写成void )

3)构造函数支持函数重载(可以有多个初始化的函数)

4)对象创建的同时编译器默认自动调用构造函数

5)有参的构造函数以及无参的构造函数书写的区别

无参构造函数 在调用的时候后面是不能添加 ()

有参构造函数在创建对象的同时就给出对应的参数即可

当用户自己没有定义构造函数的时候,编译器会默认生成一个构造函数

只不过此时是隐式调用构造函数

至于结果为什么是随机值,下面会继续讲解

6)编译器生成的默认构造函数:对对象初始化的时候为什么是随机值??

相信这是不少老铁们的疑惑

这是因为编译器底层对类型做了一些特殊的处理:

对于内置的类型(基本类型)(比如int ,char,float,double,指针等等):编译器在调用构造函数的时候不会对类的成员进行处理

但是对于那些自定义类型:struct , class ,enum, unio等等,编译器在调用构造函数的时候对这些成员会进行处理

栗子见下:

对于默认构造函数的理解:

相信有不少老铁们会认为:只有编译器生成的构造函数才是 默认构造函数,那你就大错特错了

有参的构造函数,无参的构造函数,编译器生成的构造函数都是默认构造函数

 针对编译器不能对内置类型做处理这一问题,引出了全缺省的构造函数

那么问题又来了:当无参的构造函数和全缺省的构造函数同时存在,并且对象也没有给出对应 的参数,编译器是否可以编译通过呢???

自然是不言而喻的,不能通过:因为此时发生歧义了,编译器也不知道自己到底是调用无参的构造函数还是调用全缺省的构造函数

所以:对于无参的构造函数以及全缺省的构造函数只能存在一个(一般建议:选择全缺省的构造函数)

3:析构函数

在学习数据结构链表等知识章节的时候,我们总是频繁的调用初始化函数,销毁函数,在C++里面,我们把资源的释放交给析构函数来进行处理,析构函数是和构造函数功能相反的一个函数,但是他们之间又有许多的共同点

析构函数基本特性 

 1)析构函数名字和类名一样,但是需要在类名前面加上一个 ~(方便编译器与构造函数进行区分)

2)没有返回类型也没有参数(不需要写 void )

3) 编译器会自动调用析构函数(当前对象销毁时候调用)

4)析构函数不支持函数重载

5)当用户没有显示调用析构函数的时候,编译器会在对象生命周期结束的时候自动调用(隐式调用)

6)和构造函数一样:编译器对内置类型(基本类型)的类的成员不做处理,对自定义类型的成员做处理

7) 如果类中没有申请资源时,析构函数可以不写,直接使用编译器生成的默认析构函数,比如
    有资源申请时,一定要写,否则会造成内存泄漏

关于析构函数 的基本使用如下:

编译器隐式调用Date的析构函数,显示调用Time的析构函数(用户没有定义析构函数)的栗子:

析构函数是否需要写:看情况分析:

 

4:拷贝构造函数
4.1概念:

拷贝构造函数:只有单个形参,该形参是对本类类型对象的引用(一般常用const修饰),在用已存在的类类型对象创建新对象时由编译器自动调用

4.2 特性:

1)拷贝构造函数是构造函数的一个重载

2 拷贝构造函数的参数只有一个且必须是类类型对象的引用(注:采用传值传参,会出现无限递归的情况)

3)当用户没有定义拷贝构造函数的时候编译器会调用默认 的拷贝构造函数(隐式),但是这个默认的拷贝构造函数只能对内置类型进行浅层 拷贝(值拷贝),当涉及到自定义类型的成员的时候,浅层拷贝就会有问题

4)类中如果没有涉及资源申请时,拷贝构造函数可写可不写;涉及到资源申请时,则拷贝构造函数必须写,否则编译器就是执行浅层拷贝(出现一些问题)。

拷贝构造函数的基本使用栗子:

注意以下是错误使用范例:

 

 浅层拷贝应用

深层拷贝:

注意深层拷贝不再是简简单单的拷贝那么简单了,这时候涉及到析构函数先后调用的问题

分析:

问题不仅仅是体现这方面:当Pop的时候也是有问题

 关于拷贝构造函数的综合应用场景:

class Date
{
public:
	Date(int year, int minute, int day)
	{
		cout << "Date(int,int,int):" << this << endl;
	}
	Date(const Date& d)
	{
		cout << "Date(const Date& d):" << this << endl;
	}
	~Date()
	{
		cout << "~Date():" << this << endl;
	}
private:
	int _year;
	int _month;
	int _day;
};
Date Test(Date d)
{
	Date temp(d);//用已经存在的对象d初始化temp,就需要调用拷贝构造函数
	cout << "&temp:" << &temp << endl;
	return temp;//返回类型:是一个类,就需要调用拷贝构造函数,注意此时虽然是返回一个局部变量,但是编译器是借助创建一个临时变量来返回的 
}
int main()
{
	//拷贝构造函数适应场景:1)已经存在的类对象创建一个新的类对象  2)函数返回类型是类的   3)函数参数类型:类的类型
	Date d1(2022, 1, 13);//调用构造函数创建d1这个对象,注意拷贝函数是一个特殊的构造函数(构造函数的一个重载)
	cout << "&d1:" << &d1 << endl;
	Test(d1);//传值传参,就需要调用拷贝 构造函数
	return 0;
}

对以上结果分析见下:

5:赋值运算符重载
5.1运算符重载

C++为了增强代码的可读性引入了运算符重载,运算符重载是具有特殊函数名的函数,
也具有其返回值类型,函数名字以及参数列表,其返回值类型与参数列表与普通的函数类似。
函数名字为:关键字operator后面接需要重载的运算符符号。
函数原型:返回值类型  operator操作符   (参数列表)

以实现 '>'  '<'  '<='   '>='   '=='   '!= '运算符的重载为例吧:

注意:对于bool 类型,true一般以1代表真,false以0代表假

当把重载函数定义在类里面的时候,此时默认隐藏一个参数,也就是this指针,这个指针指向左操作数

 代码:

class Date
{
public:
	int _year;
	int _month;
	int _day;
	void Print()
	{
		cout << _year << '/' << _month << '/' << _day << endl;
	}
	Date(int year = 2024, int month = 3, int day = 26)//全缺省的构造函数(对于内置类型不做处理,对于自定义类型做处理)
	{
		_year = year;
		_month = month;
		_day = day;
	}
	//注意对于Date这个类不需要写拷贝构造函数,编译器可以进行处理
	Date(const Date& d)//拷贝构造函数(对应内置类型是做浅层拷贝,自定义类型深层拷贝)
	{
		_year = d._year;
		_month = d._month;
		_day = d._day;
	}
	// d1 > d2 *this 默认指向运算符的左操数,此时d是d2的引用
	bool operator>(const Date& d)
	{
		if (_year > d._year)
		{
			return true;
		}
		else if (_year == d._year && _month > d._month)
		{
			return  true;
		}
		else if (_year == d._year && _month == d._month && _day > d._day)
		{
			return true;
		}
		else 
			return false;
	}
	// d1 == d2
	bool operator==(const Date& d)
	{
		return _year == d._year && _month == d._month && _day == d._day;
	}
	bool operator!=(const Date& d)
	{
		//直接复用之前代码
		return !(*this == d);
	}
	//d1 >= d2 ===> d1 > d2 || d1 == d2
	bool operator>=(const Date& d)
	{
		return(((*this) > d) || (*this) == d);
	}
	// d1 <= d2
	bool operator<= (const Date& d)
	{
		//对 . 进行取反
		return !(*this > d);
	}
};
int main()
{
	Date d1(2024, 11,26);//调用构造函数进行实例化
	Date d2;// 只不过是用全缺省函数进行初始化
	cout << (d1 <= d2) << endl;//编译器会自动转换成调用这个!= 运算符重载函数 d1.operator(d2)

	return 0;
}

5.2 赋值运算符重载
class Time
{
public:
	int _hour;
	int _min;
	int _sec;
	Time(int hour = 22, int min = 22, int sec = 22)//全缺省的构造函数
	{
		_hour = hour;
		_min = min;
		_sec = sec;
	}
	Time(const Time& t)//拷贝构造函数
	{
		_hour = t._hour;
		_min = t._min;
		_sec = t._sec;
	}
	void operator=(Time t)// 对 = 这个运算符重载
	{
		_hour = t._hour;
		_min = t._min;
		_sec = t._sec;
	}
	
};
int main()
{
	Time t1(22, 12, 20);
	Time t2;
	t2 = t1;
	cout << t1._hour << ':' << t1._min << ':' << t1._sec << endl;
	cout << t2._hour << ':' << t2._min << ':' << t2._sec << endl;

	return 0;
}

 

5.3日期类的实现
5.3.1实现日期+天数的函数

日期+天数 可以得到一个精确的日期

2024,3, 27 +  50 这个涉及到进位的问题(如同加法的运算一样)

此时天数是 77 显然是大于3月份的天数

进位 & 并减去是3月份的天数,

:2024 ,4 ,46,同理,大于4月份的天数

2024,5,16 此时的满足5月份的天数要求

最终结果:2024,5,16

分析:

1)写一个获取当前月份的天数的函数

2)用循环判断当前月份的天数是否满足要求

运行结果:

和网上求的一样,觉得没有啥问题

当我们连续调用 += 这个函数的时候,问题就来了

正确答案:

其实简单分析一下,就知道问题所在了:第一次调用是以 d1为参考对象进行50天后的日期推算。第二次调用虽然是以d1为参考对象,但是此时的d1不再是2024,3,27了,因为实现的+=这个运算符重载的函数改变了d1

改进之后的代码:实现 + 的运算符重载即可

代码:

Date Date::operator+=(int day)
{
	_day += day;
	while (_day > GetMonthDay(_year,_month,_day))
	{
		if (_month >= 13)
		{
			_month = 1;
			_year++;
		}
		_day -= GetMonthDay(_year, _month, _day);//减去当月的天数
		_month++;
	}
	return *this;//返回*this,每次调用都会改变当前的对象
	//注意当这样写有bug
	//如果连续调用进行日期相加的话,就出问题

}
Date Date::operator+(int day)
{
	//对上面的+=改进
	Date tmp(*this);//对*this 进行拷贝一份
	tmp += day;//直接调用 +=的函数
	return tmp;
	//弊端:涉及到拷贝的调用
}

运行结果:

对最后一个结果进行验证:

 5.3.2 日期 - 日期的函数

在实现日期- 日期函数之前,先写一个日期 - 天数的函数

这个分析过程和日期+天数一样,只不过这里是进行借位的运算:

2024,2,15 - 50

用当前月份的天数 - 50 =  - 35  < 0

此时向前借位:注意这里借的是 1月份的天数,既不是借的2月份的天数,也不是借的3月份的天数

用借来月份的天数 + 当前天数 - 35 = - 4 < 0

依然进行借位:借的的前一年,也就是 2023年的12月份的天数 31 + (-4) = 27

注意这里在代码实现上的细节:对应的年要 - -,对应的月份要跟新到12 月份

也就是最终结果:2023,12,27

 

对于这个代码其实还是有一点 的Bug,连续调用这个 - = 运算符重载函数的时候,对应 的结果会觉得比较怪:

通过代码调试就知道了:- = 这个运算符重载函数每次调用都会对当前的参考对象进行改变

优化之后的代码:

 

运行结果:

 日期 - 日期:

// d2 - d1 = 天数
int Date::operator-(const Date& d)
{
	Date max = *this;
	Date min = d;
	Date tmp = d;
	//找出较大的日期
	if (max < min)
	{
		tmp = max;
		max = min;
		min = tmp;
	}
	//找出较小的那个日期,对这个较小的日期进行++,直到与较大的日期相等
	int count = 0;
	while (min < max)
	{
		count++;
		++min;
	}
	return count;
}

 

 运行结果:

6:const成员函数
定义

const修饰的“成员函数”称之为const成员函数,const修饰类成员函数,实际修饰该成员函数隐含的this指针,表明在该成员函数中不能对类的任何成员进行修改。
 

应用

 此时定义2个Date 的对象,区别就是有无const 进行修饰

通过运行结果看出:都可以进行调用Print这个函数

思考以下几个问题

        1. const对象可以调用非const成员函数吗?
        2. 非const对象可以调用const成员函数吗?
        3. const成员函数内可以调用其它的非const成员函数吗?
        4. 非const成员函数内可以调用其它的const成员函数吗?

      1. const对象可以调用非const成员函数吗?

不能;此时会造成权限的放大
        2. 非const对象可以调用const成员函数吗?

可以;权限可以缩小
        3. const成员函数内可以调用其它的非const成员函数吗?

不能
        4. 非const成员函数内可以调用其它的const成员函数吗?

可以,权限缩小

7:取地址以及const 取地址操作符重载

注意这2个函数,对于第二个函数而言:他是一个const成员函数,返回值类型 const Date*

一般可以不写此函数,编译器会默认生成

 运行结果:

 结语:

 OK~,以上就是我今日要为各位share的,对于类这个模块的学习非常重要,而且知识点也特别零散。希各位都可以有所收获从这篇文章里面,当然也是有许多不足之处,欢迎各位大佬随时指出,谢谢支持!

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

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

相关文章

【随笔】Git 高级篇 -- 分离 HEAD(十一)

&#x1f48c; 所属专栏&#xff1a;【Git】 &#x1f600; 作  者&#xff1a;我是夜阑的狗&#x1f436; &#x1f680; 个人简介&#xff1a;一个正在努力学技术的CV工程师&#xff0c;专注基础和实战分享 &#xff0c;欢迎咨询&#xff01; &#x1f496; 欢迎大…

Stream 流和 Lambda 组装复杂父子树形结构

在最近的开发中&#xff0c;遇到了两个类似的需求&#xff1a;都是基于 Stream 的父子树形结构操作&#xff0c;返回 List 集合对象给前端。于是在经过需求分析和探索实践后有了新的认识&#xff0c;现在拿出来和大家作分享交流。 一般来说完成这样的需求大多数人会想到递归&a…

认识什么是Git

目录 1. 认识Git 1.1. 问题引入 1.2. 概念 1.3. 作用 1.4. 如何学 1.5. Git 安装 1.6. Git配置用户信息 2. Git仓库 2.1. Git 仓库&#xff08;repository&#xff09; 2.2. 创建 2.3. 需求 3. Git的三个区域 3.1. Git 使用时的三个区域 3.2. 工作区的内容&#…

11-代码随想录34在排序数组中查找元素的第一个和最后一个位置

34. 在排序数组中查找元素的第一个和最后一个位置 给定一个按照升序排列的整数数组 nums&#xff0c;和一个目标值 target。找出给定目标值在数组中的开始位置和结束位置。 如果数组中不存在目标值 target&#xff0c;返回 [-1, -1]。 进阶&#xff1a;你可以设计并实现时间…

SpringBoot参数校验@Valid 和 @Validated注解使用详解

JSR-303 是 JAVA EE 6 中的一项子规范&#xff0c;叫做 Bean Validation&#xff0c;官方参考实现是Hibernate Validator。 注意&#xff1a;JSR-303实现与 Hibernate ORM 没有任何关系。 JSR 303 用于对 Java Bean 中的字段的值进行验证。 Spring MVC 3.x 之中也大力支持 JS…

Linux :进程的程序替换

目录 一、什么是程序替换 1.1程序替换的原理 1.2更改为多进程版本 二、各种exe接口 2.2execlp ​编辑 2.2execv 2.3execle、execve、execvpe 一、什么是程序替换 1.1程序替换的原理 用fork创建子进程后执行的是和父进程相同的程序(但有可能执行不同的代码分支),子进程往…

python小项目——时钟模拟

钟表是一种计时的装置&#xff0c;也是计量和指示时间的精密仪器。钟表的样式千变万化&#xff0c;但是用来显示时间的表盘相差无几&#xff0c;大多数钟表表盘的样式由刻度&#xff08;共60个&#xff0c;围成圆形&#xff09;、指针&#xff08;时针、分针和秒针&#xff09;…

C++ 11是如何封装Thread库的?

引言 C11 标准引入了一个重要的特性&#xff0c;即原生线程支持&#xff0c;这标志着C语言在并发编程领域迈出了坚实的步伐。在此之前&#xff0c;开发人员在进行跨平台的多线程编程时&#xff0c;不得不依赖于操作系统提供的特定API&#xff0c;如Windows API或POSIX Threads…

RuoYi-Vue若依框架-集成mybatis-plus报错Unknown column ‘search_value‘ in ‘field list‘

报错信息 ### Error querying database. Cause: java.sql.SQLSyntaxErrorException: Unknown column search_value in field list ### The error may exist in com/ruoyi/sales/mapper/ZcSpecificationsMapper.java (best guess) ### The error may involve defaultParameter…

【图像分割】nnUnetV1与V2的Linux部署与应用命令

以前觉得麻烦&#xff0c;一直没用过nnunet&#xff0c;虽然知道它很火&#xff0c;最近一个契机&#xff0c;部署使用了一下nnunet&#xff0c;记录一下其部署和使用的方法与命令。 1、部署 首先&#xff0c;我有一个环境&#xff0c;这个环境可以是以前就有的&#xff0c;也可…

常州小程序案例分享:行业领军企业的数字化转型之路

随着信息技术的快速发展&#xff0c;数字化转型已成为各行各业企业提升竞争力、适应市场变革的关键战略。在江苏省常州市&#xff0c;一些行业领军企业凭借敏锐的市场洞察力和前瞻性的创新思维&#xff0c;成功借助小程序这一轻量化应用平台&#xff0c;实现了业务流程优化、用…

Kubernetes(k8s):如何进行 Kubernetes 集群健康检查?

Kubernetes&#xff08;k8s&#xff09;&#xff1a;如何进行 Kubernetes 集群健康检查&#xff1f; 一、节点健康检查1、使用 kubectl 查看节点状态2、查看节点详细信息3、检查节点资源使用情况 2、Pod 健康检查2.1、 使用 kubectl 查看 Pod 状态2.2、 查看特定 Pod 的详细信息…

基于Springboot+vue的宠物服务管理系统+论文文档

基于Springbootvue的宠物服务管理系统论文文档 预览 简介 本系统共分为三个角色&#xff1a;管理员、用户&#xff1a; 管理员&#xff1a;管理员管理、密码修改、用户管理、充值管理、商品分类管理、商品信息管理、订单信息管理、分享趣事管理、医疗服务管理、服务预约管理…

力扣刷题Days32-92. 反转链表 II(js)

1,题目 给你单链表的头指针 head 和两个整数 left 和 right &#xff0c;其中 left < right 。请你反转从位置 left 到位置 right 的链表节点&#xff0c;返回 反转后的链表 。 2&#xff0c;代码 一次遍历「穿针引线」反转链表&#xff08;头插法&#xff09; /*** Defi…

LeetCode 96. 不同的二叉搜索树

给你一个整数 n &#xff0c;求恰由 n 个节点组成且节点值从 1 到 n 互不相同的 二叉搜索树 有多少种&#xff1f;返回满足题意的二叉搜索树的种数。 示例 1&#xff1a; 输入&#xff1a;n 3 输出&#xff1a;5示例 2&#xff1a; 输入&#xff1a;n 1 输出&#xff1a;1提…

【蓝桥杯嵌入式】六、真题演练(一)-1演练篇:第 14 届真题

温馨提示&#xff1a; 真题演练分为模拟篇和研究篇。本专栏的主要作用是记录我的备赛过程&#xff0c;我打算先自己做一遍&#xff0c;把遇到的问题和不同之处记录到演练篇&#xff0c;然后再返回来仔细研究一下&#xff0c;找到最佳的解题方法记录到研究篇。 目录 解题记录&…

2023年EI会议论文已见刊/检索进展汇总

2023年录用的会议论文已在SPIE、ACM、IEEE等出版社正式上线见刊&#xff0c;并已陆续完成EI Compendex数据库收录&#xff0c;详情如下&#xff1a; EIECT 2023——IEEE出版&#xff0c;并完成EI收录 会议信息&#xff1a; 第三届电子信息工程与计算机技术国际学术会议&…

MapReduce [OSDI‘04] 论文阅读笔记

原论文&#xff1a;MapReduce: Simplified Data Processing on Large Clusters (OSDI’04) 1. Map and Reduce Map&#xff1a;处理键值对&#xff0c;生成一组中间键值对Reduce&#xff1a;合并与同一中间键相关的所有中间值process overview&#xff1a;分割输入数据&#x…

EF数据持久化(三层架构,公司查,改)

效果图 Model设置具体流程在下面链接中 https://blog.csdn.net/Mr_wangzu/article/details/136805824?spm1001.2014.3001.5501 DAL using System; using System.Collections.Generic; using System.Linq; using System.Web; using WebApplication2.Models; namespace WebAppli…

力扣由浅至深 每日一题.20 环形链表

山穷水尽&#xff0c;柳暗花明 —— 24.4.3 环形链表 给你一个链表的头节点 head &#xff0c;判断链表中是否有环。 如果链表中有某个节点&#xff0c;可以通过连续跟踪 next 指针再次到达&#xff0c;则链表中存在环。 为了表示给定链表中的环&#xff0c;评测系统内部使用整…