【C++】--类和对象(3)

news2025/1/10 21:30:09

🤑个人主页: 起名字真南
🤑个人专栏:【数据结构初阶】 【C语言】 【C++】

请添加图片描述

目录

  • 1 深入构造函数
  • 2 类型转换
  • 3 static成员
  • 4 友元函数
  • 5 内部类
  • 6 匿名对象

1 深入构造函数

  • 之前我们实现构造函数的时候,初始化成员变量都是在函数体内赋值,构造函数初始化还有另一个方式,就是初始化列表,初始化列表的使用方式是从一个冒号开始,成员变量之间用‘ ,’(逗号)分隔开,每个成员变量的后面都有一个括号里面存放着初始化的值或表达式。
  • 每个成员变量都只能在初始化列表中出现一次,可以理解为是每个成员变量进行定义初始化的地方。
  • 引用成员变量,const成员变量,没有默认构造的类类型成员变量,必须放在初始化列表的地方进行初始化,否则会编译报错
  • C++11支持在成员变量声明的地方给缺省值,这个缺省值目的是给没有显示在初始化列表初始化的变量使用的。
  • 使用初始化列表进行初始化,因为即便没有在初始化列表进行初始化编译器同样会经过初始化列表,如果在变量定义的时候给了缺省值那么就会用缺省值进行初始化,如果没有缺省值就会按照括号里的值或表达式进行初始化,对于没有在初始化列表显示的内置类型是否进行初始化却决于编译器,而对于自定义类型则会调用他的默认构造函数,如果没有默认构造就会造成编译报错。
  • 初始化列表的初始化顺序与该成员变量在初始化列表中的顺序无关,至于这个变量的声明顺序有关,先声明的先初始化。
    总结 :
    无论是否写初始化列表,每个构造函数都有初始化列表。
    无论是否在初始化列表显示初始化,每个成员变量都会走一遍初始化列表。

在这里插入图片描述

#include<iostream>
using namespace std;

class Time
{
public:
	Time(int hour)
		:_hour(hour)
	{
		cout << "Time()" << endl;
	}
private:
	int _hour;
};
class Date
{
public:
	Date(int& x, int year = 1, int month = 1, int day = 1)
		:_year(year)
		, _month(month)
		, _day(day)
		, _t(12)
		, _ret(x)
		, _n(1)
	{
	//只能在初始化列表初始化,不能再函数体内进行初始化
		//_ret = x;
		//_t = 12;
		//_n = 1;
	}
	void Print()
	{
		cout << _year << "-" << _month << "-" << _day << endl;
	}
private:
	int _year;
	int _month;
	int _day;
	// 以下变量能且只能在初始化列表进行初始化
	Time _t;       //类类型成员变量   没有默认构造
	int& _ret;     //引用成员变量   
	const int _n;  //const类型成员变量
};
int main()
{
	int i = 0;
	Date d1(i);
	d1.Print();
	return 0;
}

运行结果如下:
在这里插入图片描述
虽然Time类型的成员变量都是内置类型成员变量,并且在初始化的时候可以不写默认构造但是由于在Time中已经写了一个构造函数所以不会生成默认构造,而在Date类中作为了自定义类类型的成员变量如果没有在初始化列表中初始化就会调用他的默认构造,因为它没有默认构造就会报错。
如果把初始化列表中的Time删除就会出现下面的错误:
在这里插入图片描述

class Time
{
public:
	Time(int hour)
		:_hour(hour)
	{
		cout << "Time()" << endl;
	}
private:
	int _hour;
};
class Date
{
public:
	Date()
		:_month(1)
	{
		cout << "Date()" << endl;
	}
	void Print()
	{
		cout << _year << "-" << _month << "-" << _day << endl;
	}
private:
	int _year = 1;
	int _month = 1;
	int _day;
	// 注意这里不是初始化而是给初始化列表提供了缺省值
	//如果初始化列表没有显示初始化,就是用这个缺省值初始化
	Time _t = 1;       
	int* _ret = (int*)malloc(sizeof(int) * 3);      
	const int _n = 1; 
};
int main()
{
	Date d1;
	d1.Print();
	return 0;
}

运行结果如下:
在这里插入图片描述

2 类型转换

  • C++支持内置类型隐式类型转换为类类型对象,需要有相关内置类型作为参数的构造函数
  • 构造函数前面加上explicit就不再支持隐式类型转换
  • 类类型对象之间也可以隐式类型转换,需要相对应的构造函数支持
#include<iostream>
using namespace std;

class A
{
public:

	//构造函数explicit就不再支持隐式类型转换
	// explicit A(int a1)

	A(int a1)
		:_a1(a1)
	{}
	A(int a1, int a2)
		:_a1(a1)
		,_a2(a2)
	{}
	void print()
	{
		cout << _a1 << " " << _a2 << endl;
	}
	int get() const
	{
		return _a1 + _a2;
	}
private:
	int _a1 = 1;
	int _a2 = 2;
};
class B
{
public:
	B(const A& a)
		:_b(a.get())
	{}
private:
	int _b = 0;
};
int main()
{
	A aa1 = 1;
	aa1.print();

	const A& aa2 = 1;
	//多参数传参用大括号
	A aa3 = { 3, 3 };
	aa3.print();
	cout << aa3.get() << endl;
	//A 类型 隐式转换为 B类型
	B b = aa3;

	const B& rb = aa3;
	return 0;
}

3 static成员

  • 用static修饰的成员变量叫做静态成员变量,必须要在类外进行初始化
  • 静态成员变量为所有类对象共享,不属于某个具体的对象,不存在对象中,存放在静态区中。
  • 用static修饰的成员函数称为静态成员函数,静态成员函数没有this指针。
  • 静态成员函数中可以访问其他的静态成员,但是不能访问非静态成员因为没有this指针。
  • 非静态成员函数可以访问任意的静态成员变量和静态成员函数
  • 突破类域就可以访问静态成员,可以通过类名::静态成员,对象.静态成员来访问静态成员变量和静态成员函数。
  • 静态成员也受public,private,protected等访问限定符限制。
  • 静态成员函数不能在声明的位置给缺省值进行初始化,因为缺省值是给构造函数初始化列表使用的,静态成员不属于某个对象,不走初始化列表。
#include<iostream>
using namespace std;

class A
{
public:
	A()
	{
		++_scount;
	};
	A(const A& t)
	{
		++_scount;
	}
	~A()
	{
		--_scount;
	}
	static int GetCount()
	{
		return _scount;
	}
private:
	//类里面声明
	static int _scount;
};

//类外面初始化
int A::_scount = 0;

int main()
{
	cout << A::GetCount() << endl;
	A a1, a2;
	cout << A::GetCount() << endl;
	A a3(a1);
	cout << A::GetCount() << endl;
	cout << a3.GetCount() << endl;
	return 0;
}

运行结果:
在这里插入图片描述
第一次输出的结果是0,是因为我们在初始化的时候初始化的值为0,然后在每一次调用构造函数的时候就会加1,调用析构的时候减1,我们可以看到构造了a1,a2两个变量所以调用了两次++,然后在使用拷贝构造a3,调用构造函数在++所以这个时候的count是3,每次调用构造函数都会++,然后最后函数销毁时会自动调用析构函数,调用三次。

4 友元函数

  • 友元提供了一种突破类域访问限制的方法,友元分为:友元函数,友元类,在函数声明和类声明前面加上friend,并且把友元声明放到一个类里面。
  • 外部友元函数可以访问内部类的私有和保护变量,友元函数仅仅是一种声明并不是类的成员函数。
  • 友元函数可以在类定义的任何地方声明不受访问限定符的限制。
  • 一个函数可以是多个类的友元函数
  • 友元类的关系是单向的并不是双向的比如A是B的友元,但是B不是A的友元
  • 友元的关系不能传递,比如A是B的友元,B是C的友元,但A不是C的友元
  • 容易破坏封装不宜多用
#include<iostream>
using namespace std;

class A; //前置声明为了调用函数的时候可以找到A类
class B
{
	friend void func(const A& aa, const B& bb);

private:
	int _b1 = 1;
	int _b2 = 2;
};

class A
{
	friend void func(const A& aa, const B& bb);


private:
	int _a1 = 1;
	int _a2 = 2;
};

void func(const A& aa, const B& bb)
{
	cout << aa._a1 << endl;
	cout << bb._b1 << endl;
}
int main()
{
	A a1, a2;
	B b1, b2;
	func(a1, b1);
	return 0;
}

在写上面的代码我们需要注意一个情况就是关于前置声明,因为编译器只会向上查找,在B类中的友元函数用到了A类所以我们需要在B类的前面提前声明A类告诉编译器A类存在。

class C
{
	friend class D;
private:
	int _c1 = 1;
	int _c2 = 2;
};

class D
{
public:
	void func01(const C& cc)
	{
		cout << cc._c1 << endl;
		cout << _d1 << endl;
	}
	void func02(const C& cc)
	{
		cout << cc._c2 << endl;
		cout << _d2 << endl;
	}

private:
	int _d1 = 3;
	int _d2 = 4;
};

int main()
{
	C cc;
	D dd;
	dd.func01(cc);
	dd.func02(cc);

	return 0;
}

5 内部类

  • 如果一个类定义在一个类的内部,那么这个内部的类就叫做内部类,内部类是一个独立的类,跟定义在全局相比他只受外部类的类域和访问限定符限制,所以外部类定义的对象中不包含内部类。
  • 内部类默认是外部类的友元
  • 内部类本质上也是一种封装,如果A类和B类有紧密的联系,并且A类设计出来是专门为B类使用的那么就可以把B类设计为A类的内部类,B作为内部类可以访问A类的私有成员变量。
class AA
{
public:
	class BB
	{
	public:
		void func01(const AA& aa)
		{
			cout << _k << endl;
			cout << aa._x << endl;
		}
	};
private:
	static int _k;
	int _x = 1;
};

int AA:: _k = 2;

int main()
{
	cout << sizeof(AA) << endl;

	AA aa;
	AA::BB bb;

	bb.func01(aa);

	return 0;
}

结果展示:
在这里插入图片描述
这里sizeof的大小是4的原因是_k是在静态区不算做A类里面

6 匿名对象

  • 用类型(实参)定义出来的对象叫做匿名对象,用类型 对象名 定义出来的对象叫做有名对象
  • 匿名对象的生命周期只在当前一行。
class AAA
{
public:
	   AAA(int a = 0)
			   :_a(a)
	   {
			   cout << "AAA(int a)" << endl;
	   }
	   ~AAA()
	   {
			   cout << "~AAA()" << endl;
	   }
private:
	   int _a;
};
class Solution {
public:
	int Sum_Solution(int n) {
		//...
		return n;
	}
};
int main()
{
	//匿名对象的定义
	AAA();
	AAA(1);
	//有名对象
	AAA a2(2);

	Solution().Sum_Solution(10);
	return 0;
}

运行结果:
在这里插入图片描述
这里分别调用了三次构造三次析构证明对象确实存在生命周期只存在一行在下一个对象创建之后就已经销毁调用了析构

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

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

相关文章

linux-冯诺伊曼体系结构以及操作系统

冯诺依曼体系结构 我们不畅见到计算机&#xff0c;如笔记本&#xff0c;不常见的如服务器&#xff0c;大部分都遵循着冯诺伊曼体系结构 截至目前&#xff0c;我们所认识的计算机&#xff0c;都是由一个个硬件组件组成。 输入单元&#xff1a;包括键盘 , 鼠标&#xff0c;扫描…

鸿蒙开发之ArkUI 界面篇 二十四 计数器案例

计数器案例&#xff0c;点击’-‘按钮&#xff0c;数字减少1&#xff0c;点击啊‘’按钮&#xff0c;数字加一 分析&#xff1a;这里需要三个组件&#xff0c;外层容器是Row&#xff0c;从左往右的组件分别是ButtonTextButton&#xff0c;涉及到修改更新界面&#xff0c;变量需…

Java语言教程:打造你的第一款五子棋游戏 JAVA学习攻略心得总结

目录 Java语言教程&#xff1a;打造你的第一款五子棋游戏 游戏简介 游戏代码 自学Java攻略及功能 1. Java基础知识 2. 面向对象编程&#xff08;OOP&#xff09; Java简介 游戏简介 五子棋&#xff0c;又称连珠&#xff0c;是一种两人对弈的棋类游戏。五子棋的规则简单易…

红帽7—Mysql路由部署

MySQL Router 是一个对应用程序透明的InnoDB Cluster连接路由服务&#xff0c;提供负载均衡、应用连接故障转移和客户端路 由。 利用路由器的连接路由特性&#xff0c;用户可以编写应用程序来连接到路由器&#xff0c;并令路由器使用相应的路由策略 来处理连接&#xff0c;使其…

添加菜品到购物车

分析 数据库设计 代码开发 三个步骤&#xff1a; 判断当前商品是否已经在购物车中如果在购物车中&#xff0c;更新购物车中商品数量如果不在购物车中&#xff0c;添加到购物车 controller层 /*** 添加购物车** return*/PostMapping("/add")ApiOperation("添加购…

MediaTek携手“天玑芯世界探索官”辛芷蕾,开启科技新世界

2024年10月9日&#xff0c;MediaTek举办 2024天玑旗舰芯片新品发布会&#xff0c;正式发布旗舰 5G 智能体 AI 芯片天玑 9400。发布会上&#xff0c;MediaTek回顾了天玑品牌5周年的辉煌历程&#xff0c;展望了AI技术将推动人类进入一个充满人文关怀和情感交流的智慧科技新世界&a…

智能优化算法-引力搜索优化算法(GSA)(附源码)

目录 1.内容介绍 2.部分代码 3.实验结果 4.内容获取 1.内容介绍 引力搜索优化算法 (Gravitational Search Algorithm, GSA) 是一种基于牛顿万有引力定律的元启发式优化算法&#xff0c;由Rashedi等人于2009年提出。GSA通过模拟天体之间的引力作用来搜索最优解&#xff0c;适用…

.NET周刊【9月第4期 2024-09-22】

国内文章 .NET常见的几种项目架构模式&#xff0c;你知道几种&#xff1f;&#xff08;附带使用情况投票&#xff09; https://www.cnblogs.com/Can-daydayup/p/18421685 文章介绍了软件项目的几种架构模式&#xff0c;包括三层架构、MVC、DDD、整洁架构和CQRS。这些模式通过…

算法:双指针系列(二)——对撞指针

双指针系列&#xff08;二&#xff09;——对撞指针 &#xff08;一&#xff09;盛水最多的容器&#xff08;一&#xff09;题目分析&#xff08;二&#xff09;代码展示 二、有效的三角形个数&#xff08;一&#xff09;题目分析&#xff08;二&#xff09;代码展示 三、查找总…

Scalable TCP 如何优化长肥管道

来看一个极简的拥塞控制实现 net/ipv4/tcp_scalable.c&#xff0c;去掉注释不到 50 行代码。它的介绍在 Scalable TCP-improving performance in highspeed networks。由于太简单&#xff0c;估计没什么人会在意。 本文说一下它背后的道理。 无论 bic/cubic&#xff0c;westw…

从0开始linux(10)——进程(2)进程属性

欢迎来到博主的专栏&#xff1a;从0开始linux 博主ID&#xff1a;代码小豪 文章目录 进程与操作系统并发与并行进程的状态 linux的进程状态 进程与操作系统 进程的状态可分为3种&#xff0c;分别是新建态&#xff0c;运行态&#xff0c;终止态&#xff0c;就绪态与阻塞态。状态…

【深度学习总结】热力图-Grad-CAM使用

Grad-CAM使用 介绍 Grad-CAM&#xff0c;全称为Gradient-weighted Class Activation Mapping&#xff0c;是一种用于深度学习模型可视化的技术&#xff0c;特别是在卷积神经网络&#xff08;CNN&#xff09;中。它通过生成热力图来展示模型在做出决策时关注的区域&#xff0c…

反应香精市场报告:预计2030年全球市场规模将达到264.3亿美元

“反应香精”通常是指通过在食品或饮料加工过程中发生的物理、化学或酶反应而产生的风味剂。可以有意添加这些香料以增强最终产品的味道、香气或其他感官方面。它们通常用于食品和饮料行业&#xff0c;以保持一致性、提高适口性或创造独特的风味特征。生产工艺香料的方法有多种…

[论文阅读] DVQA: Understanding Data Visualizations via Question Answering

原文链接&#xff1a;http://arxiv.org/abs/1801.08163 启发&#xff1a;没太读懂这篇论文&#xff0c;暂时能理解的就是本文提出了一个专门针对条形图问答的数据集DVQA以及一个端到端模型SANDY&#xff0c;模型有两个版本&#xff0c;Oracle和OCR。主要解决的问题是固定词表无…

树莓派3b安装ubuntu18.04服务器系统server配置网线连接

下载ubuntu镜像网址 img镜像&#xff0c;即树莓派官方烧录器使用的镜像网址 ubuntu18.04-server&#xff1a;ARM/RaspberryPi - Ubuntu Wiki 其他版本&#xff1a;Index of /ubuntu/releases 下载后解压即可。 发现使用官方烧录器烧录配置时配置wifi无论如何都不能使用&am…

AI的历史、现状与理论基础

在本篇文章中&#xff0c;我们将深入探讨人工智能&#xff08;AI&#xff09;的起源、现状以及理论基础&#xff0c;为读者提供一个全面的理解框架。 I. 引言 人工智能&#xff08;AI&#xff09;作为一门跨学科的研究领域&#xff0c;其目标是模拟、延伸和扩展人的智能。本文…

是德(KEYSIGHT) N9040A、N9040B 信号分析仪

Keysight N9040B 的特性和规格包括&#xff1a; 功能性 3 Hz 至 8.4、13.6 或 26.5 GHz&#xff1b;使用是德科技智能混频器将频率扩展至 110 GHz&#xff0c;使用其他供应商的混频器将频率扩展至 THz10 MHz&#xff08;标准&#xff09;、25、40、255 或 510 MHz 分析带宽全…

第十八篇:一文说清楚ICMP的底层原理

作为程序员或者网络工程师&#xff0c;有时候无法访问对方主机&#xff1b;导致这个现象的有很多原因&#xff0c;那要排查具体的网络原因&#xff0c;可能会用到ping的指令。而ping的底层实现是互联⽹控制报⽂协议&#xff08;ICMP&#xff09;。 ICMP 全称是 Internet Contr…

清华系“仓颉”来袭:图形起源:用AI颠覆字体设计,推动大模型商业化落地

大模型如何落地&#xff1f;又该如何实现商业化&#xff1f;这一议题已成为今年科技领域的焦点话题。 在一个鲜为人知的字体设计赛道上&#xff0c;清华创业公司“图形起源”悄然实现了商业变现&#xff1a;他们帮助字体公司将成本降低了80%&#xff0c;生产速度提升了10倍以上…

网站优化门槛低了还是高了?

自从2015年刚接触网站时&#xff0c;从一无所知到现在无人指导&#xff0c;一直跌跌撞撞走过来&#xff0c;当年花了1500元找了广东一个网友用织梦CMS做了一个门户网站&#xff0c;记得那时一星期没下楼&#xff0c;把网站折腾的千疮百孔&#xff0c;而终逐步熟悉网站建设与搜索…