C++之类和对象的中篇

news2024/11/24 13:25:17

𝙉𝙞𝙘𝙚!!👏🏻‧✧̣̥̇‧✦👏🏻‧✧̣̥̇‧✦ 👏🏻‧✧̣̥̇: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的,对于类这个模块的学习非常重要,而且知识点也特别零散。希各位都可以有所收获从这篇文章里面,当然也是有许多不足之处,欢迎各位大佬随时指出,谢谢支持!

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

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

相关文章

函数参数缺省和内联函数【C++】

文章目录 函数参数缺省函数参数缺省的条件和要求 内联函数内联函数的工作原理内联函数的定义方法内联函数的要求解决方法&#xff1a;直接在.h中定义内联函数的函数体 内联函数再Debug模式下默认是不展开的 函数参数缺省 顾名思义&#xff1a;可以少传一个/多个参数给函数&…

归并排序和分治

归并排序 归并排序是利用归并的思想实现的排序方法&#xff0c;该算法采用经典的分治策略&#xff08;分治法将问题分成一些小的问题然后递归求解&#xff0c;而治的阶段则将分的阶段得到的各答案"修补"在一起&#xff0c;即分而治之)。 分而治之 可以看到这种结构…

前端实现菜单搜索搜索(功能模版)

目录 前言正文 前言 总体界面如下所示&#xff1a; 正文 <template><div class"avue-searchs"click.self"handleEsc"><div class"avue-searchs__title">菜单搜索</div><div class"avue-searchs__content"…

游戏运营分析:如何在新游戏上线初期实现精细化运营?

一、背景介绍 在当今的手游市场中&#xff0c;每一款新游戏的发布都如同踏上一段充满未知与挑战的探险之旅。游戏刚上线时&#xff0c;运营情况往往如同飘摇的小船&#xff0c;随时可能受到风浪的侵袭。此时&#xff0c;如何准确地找到问题所在&#xff0c;为游戏的健康运营和持…

瀚海贫者福,铜子恣意游

上学时打饭追求性价比的习惯一直不改&#xff0c;半个大鱼头三块钱&#xff0c;一份豆腐一块钱&#xff0c;还有一个红烧茄子2块5&#xff0c;再加三毛钱的饭&#xff0c;共6块8毛钱&#xff0c;早晚餐也会有这类性价比高又营养的选择&#xff0c;科大食堂现在越来越人性化&…

蓝桥杯练习——拼出一个未来

选中 index.html 右键启动 Web Server 服务&#xff08;Open with Live Server&#xff09;&#xff0c;让项目运行起来。接着&#xff0c;打开环境右侧的【Web 服务】&#xff0c;就可以在浏览器中看到如下效果&#xff1a; 目标 完善 js/index.js 的 TODO 部分&#xff0c;实…

零基础学会Python

⭐简单说两句⭐ ✨ 正在努力的小新~ &#x1f496; 超级爱分享&#xff0c;分享各种有趣干货&#xff01; &#x1f469;‍&#x1f4bb; 提供&#xff1a;模拟面试 | 简历诊断 | 独家简历模板 &#x1f308; 感谢关注&#xff0c;关注了你就是我的超级粉丝啦&#xff01; &…

C 回调函数的两种使用方法

对回调&#xff08;callback&#xff09;函数的一点粗陋理解&#xff0c;在我小时候&#xff0c;隔壁村有家月饼小作坊&#xff08;只在中秋那段时间手工制作一些月饼出售&#xff0c;后来好像不做了&#xff09;&#xff0c;做出的月饼是那种很传统很经典的款式&#xff0c;里…

电机的工作电流怎么计算?

电机的工作电流计算通常需要考虑多个因素&#xff0c;包括电机的额定功率、工作电压、效率以及负载情况等。以下是一个基本的计算方法&#xff0c;用于估算直流电机或交流电机在特定条件下的工作电流。 了解电机参数 额定功率 (P_rated) 电机的额定功率是指在额定工作条件下&am…

深入C语言:探究static关键字的奥秘

文章目录 一、链接属性二、static变量1、定义静态局部变量2、在函数内部使用静态变量3、函数中静态局部变量与递归 三、static变量与全局变量的区别1、存储期与生命周期2、可见性与作用域3、使用场景4、静态与动态内存分配 注意事项 当用于不同的上下文环境时&#xff0c; sta…

005 高并发内存池_CentralCache设计

​&#x1f308;个人主页&#xff1a;Fan_558 &#x1f525; 系列专栏&#xff1a;高并发内存池 &#x1f339;关注我&#x1f4aa;&#x1f3fb;带你学更多知识 文章目录 前言本文重点一、构建CentralCache结构二、运用慢开始反馈调节算法三、完成向CentralCache中心缓存申请四…

Netty经典32连问

文章目录 1、Netty是什么&#xff0c;它的主要特点是什么&#xff1f;2、Netty 应用场景了解么&#xff1f;3、Netty 核心组件有哪些&#xff1f;分别有什么作用&#xff1f;4、Netty的线程模型是怎样的&#xff1f;如何优化性能&#xff1f;5、EventloopGroup了解么?和 Event…

第十三届蓝桥杯大赛软件赛省赛CC++大学B组

第十三届蓝桥杯大赛软件赛省赛CC 大学 B 组 文章目录 第十三届蓝桥杯大赛软件赛省赛CC 大学 B 组1、九进制转十进制2、顺子日期3、刷题统计4、修建灌木5、x进制减法6、统计子矩阵7、积木画8、扫雷9、李白打酒加强版10、砍竹子 1、九进制转十进制 计算器计算即可。2999292。 2、…

RD55UP06-V 三菱iQ-R系列C语言功能模块

RD55UP06-V 三菱iQ-R系列C语言功能模块 RD55UP06-V用户手册&#xff0c;RD55UP06-V功能&#xff0c;RD55UP06-V系统配置 RD55UP06-V参数规格&#xff1a;10BASE-T/100BASE-TX/1000BASE-T 1通道&#xff1b;字节存储次序格式小端模式; 可使用SD存储卡插槽&#xff1b;工作RAM 1…

路由、插槽

路由 前端路由&#xff1a;Hash地址(url中#后面的部分)与组件之间的对应关系 页面效果&#xff1a;在浏览器中访问不同的Hash地址时&#xff0c;会显示不同的组件 SPA项目(单页面应用程序&#xff0c;就是Vue项目&#xff0c;最后所有模板都展示在一个html上) vue路由(vue-r…

VUE3——生命周期

Vue3.0中可以继续使用Vue2.x中的生命周期钩子&#xff0c;但有有两个被更名&#xff1a; beforeDestroy改名为 beforeUnmountdestroyed改名为 unmounted Vue3.0也提供了 Composition API 形式的生命周期钩子&#xff0c;与Vue2.x中钩子对应关系如下&#xff1a; beforeCreate&g…

3D Gaussian Splatting Linux端部署指南(含Linux可视化)

3D Gaussian Splatting Linux端部署指南 目录 项目地址 部署记录 11. Linux端在线远程可视化训练进程 准备自己的数据 SIBR_remoteGaussian在线远程可视化 补充&#xff1a;sibr_3Dgaussian离线可视化训练好的模型 朋友浩哥说环境是最难配的&#xff0c;配好环境&#x…

Tinymce富文本编辑器二次开发电子病历时解决的bug

前言 本文是在Tinymce富文本编辑器添加自定义toolbar&#xff0c;二级菜单&#xff0c;自定义表单&#xff0c;签名的基础之上进行一些bug记录&#xff0c;功能添加&#xff0c;以及模版的应用和打印 项目描述 建立电子病历模版—录入&#xff08;电子病历模版和电子病历打印…

运筹学基础(三):求解整数规划的切平面法(cutting plane method)

文章目录 算法思想一个例子参考文档 算法思想 先将整数规划问题松弛为线性规划问题&#xff0c;然后割掉线性规划问题可行域的一部分&#xff08;只包含非整数解&#xff09;&#xff0c;使得线性规划问题的最优解在原整数规划问题的可行域某顶点上取得。 因此&#xff0c;割平…