C++——类和对象(下)

文章目录

  • 一、再探构造函数——初始化列表
  • 二、 类型转换
  • 三、static成员
    • 静态成员变量
    • 静态成员函数
  • 四、 友元
    • 友元函数
    • 友元类
  • 五、内部类
  • 六、匿名对象

一、再探构造函数——初始化列表

之前我们实现构造函数时,初始化成员变量主要使⽤函数体内赋值,构造函数初始化还有⼀种⽅式,就是初始化列表。

初始化列表的使⽤⽅式是以⼀个冒号开始,接着是⼀个以逗号分隔的数据成员列表,每个"成员变量"后⾯跟⼀个放在括号中的初始值或表达式。

class Date
{
public:
	//初始化列表
	Date(int year)
		:_year(year),_month(12),_day(day + 12)
	{}
private:
	int _year;
	int _month;
	int _day;
};
  • 每个成员变量在初始化列表中只能出现⼀次,语法理解上初始化列表可以认为是每个成员变量定义初始化的地⽅。
  • 引⽤成员变量,const成员变量,没有默认构造函数的类类型成员,必须放在初始化列表位置进⾏初始化,否则会编译报错。
class A
{
private:
	A(int n)
	{
		val = n;
	}
public:
	int val;
};


class Date
{
public:
	//初始化列表
	Date(int year, int month, int day)
		//定义
		:_year(year), _month(month), _day(day)
	{}
private:
	//声明
	int _year;
	int _month;
	int _day;
	const int _a;//error
	int& _b;//error
	A _c;//error
};

在这里插入图片描述

  • 初始化列表中按照成员变量在类中声明顺序进⾏初始化,跟成员在初始化列表出现的的先后顺序⽆关。建议声明顺序和初始化列表顺序保持⼀致。

  • C++11⽀持在成员变量声明的位置给缺省值,这个缺省值主要是给没有显⽰在初始化列表初始化的成员使⽤的。

#include <iostream>
using namespace std;

class Date
{
public:
	//初始化列表
	Date(int year, int month)
		//定义
		:_year(year), _month(month)
	{}
	void print()
	{
		cout << _year << "年" << _month << "月" << _day << "日" << endl;
	}
private:
	//声明
	int _year;
	int _month;
	int _day = 16;
};

int main()
{
	Date d1(2024, 7);
	d1.print();
	return 0;
}

在这里插入图片描述
总结:尽量使⽤初始化列表初始化,因为那些你不在初始化列表初始化的成员也会⾛初始化列表。
在这里插入图片描述

二、 类型转换

C++ 允许从内置类型(如 int, float 等)到类类型的隐式类型转换,但这要求类中必须定义一个接受相应内置类型参数的构造函数。
为了防止不必要的隐式转换,可以在构造函数声明前使用 explicit 关键字,这样编译器就不会自动执行这类转换,从而避免潜在的错误和混淆。

#include <iostream>
using namespace std;

class A
{
public:
	A(int n)
	{
		val = n;
	}
	A(int n, int m)
	{
		val = n;
		exc = m;
	}
private:
	int val;
	int exc;
};

int main()
{
	//1构造一个A的临时对象,再用这个临时对象拷⻉构造a
	A a = 1;
	// C++11之后才支持多参数转化
	A aa3 = { 2,2 };
	return 0;
}

加上explicit之后,编译就会报错,不能隐式类型转换了

#include <iostream>
using namespace std;

class A
{
public:
	explicit A(int n)
	{
		val = n;
	}
	explicit A(int n, int m)
	{
		val = n;
		exc = m;
	}
private:
	int val;
	int exc;
};

int main()
{
	//1构造一个A的临时对象,再用这个临时对象拷⻉构造a
	A a = 1;
	// C++11之后才支持多参数转化
	A aa3 = { 2,2 };
	return 0;
}

在这里插入图片描述

三、static成员

静态成员变量

⽤static修饰的成员变量,称之为静态成员变量,静态成员变量⼀定要在类外进⾏初始化。

  • 静态成员变量为所有类对象所共享,不属于某个具体的对象,不存在对象中,存放在静态区。
#include <iostream>
using namespace std;

class A
{
public:
	//A()
	//	// error C2438: “exc”: 无法通过构造函数初始化静态类数据
	//	:val(1),exc(2)
	//{
	//	
	//}
	void print()
	{
		cout << val << " " << exc << endl;
	}
private:
	int val;
	//类里面声明
	static int exc;
};

//类外面初始化
int A::exc = 1;

int main()
{
	A a;
	a.print();
	return 0;
}

静态成员函数

⽤static修饰的成员函数,称之为静态成员函数,静态成员函数没有this指针。

  • 静态成员函数中可以访问其他的静态成员,但是不能访问⾮静态的,因为没有this指针。
  • ⾮静态的成员函数,可以访问任意的静态成员变量和静态成员函数。
#include <iostream>
using namespace std;

class A
{
public:
	static void PK();
	void print()
	{
		cout <<  exc << endl;
	}
private:
	//类里面声明
	static int exc;
};

//类外面初始化
int A::exc = 1;

void A::PK()
{
	//error C2597: 对非静态成员“A::val”的非法引用
	//val = 1;
	cout << "666" << endl;
}

int main()
{
	A a;
	a.print();
	a.PK();
	A::PK();
	return 0;
}

在这里插入图片描述

注意:

突破类域就可以访问静态成员,可以通过类名::静态成员或者对象. 静态成员来访问静态成员变量和静态成员函数。

静态成员也是类的成员,受public、protected、private访问限定符的限制。

静态成员变量不能在声明位置给缺省值初始化,因为缺省值是个构造函数初始化列表的,静态成员变量不属于某个对象,不⾛构造函数初始化列表。

就好比这个题目:求1+2+3+…+n

class Sum
{
public:
    Sum()
    {
        num++;
        k += num;
    }
    static int GetK()
    {
        return k;
    }
private:
    static int num;
    static int k;
};

int Sum::num = 0;
int Sum::k = 0;

class Solution
{
public:
    int Sum_Solution(int n)
    {
        Sum a[n];
        return Sum::GetK();
    }
};

四、 友元

友元提供了⼀种突破类访问限定符封装的⽅式,友元分为:友元函数和友元类。
实现方法:在函数声明或者类声明的前⾯加friend,并且把友元声明放到⼀个类的⾥⾯。

友元函数

  • 外部友元函数可访问类的私有和保护成员,友元函数仅仅是⼀种声明,他不是类的成员函数。
  • 友元函数可以在类定义的任何地⽅声明,不受类访问限定符限制。
  • ⼀个函数可以是多个类的友元函数。
#include <iostream>
using namespace std;

class A
{
public:
	//友元声明
	friend void PK(A& ra);
	void print()
	{
		cout << _a << endl;
	}
private:
	int _a = 1;
};

void PK(A& ra)
{
	cout << ra._a << endl;
}

int main()
{
	A a;
	PK(a);
	a.print();
	return 0;
}

在这里插入图片描述

友元类

  • 友元类中的成员函数都可以是另⼀个类的友元函数,都可以访问另⼀个类中的私有和保护成员。
  • 友元类的关系是单向的,不具有交换性,⽐如A类是B类的友元,但是B类不是A类的友元。
  • 友元类关系不能传递,如果A是B的友元,B是C的友元,但是A不是C的友元。
#include <iostream>
using namespace std;

class A
{
public:
	//友元声明
	friend class B;
	void print()
	{
		cout << _a << endl;
	}
private:
	int _a = 1;
};

class B
{
public:
	void print(A& ra)
	{
		ra._a = 3;
		ra.print();
		cout << _b << endl;
	}
private:
	int _b = 2;
};

int main()
{
	A a;
	B b;
	b.print(a);
	return 0;
}

在这里插入图片描述

注意:友元虽然有时提供了便利,但是友元会增加耦合度,破坏了封装,所以友元不宜多⽤。

五、内部类

如果⼀个类定义在另⼀个类的内部,这个内部类就叫做内部类。

内部类是⼀个独⽴的类,跟定义在全局相⽐,他只是受外部类类域限制和访问限定符限制,所以外部类定义的对象中不包含内部类。

内部类默认是外部类的友元类。

内部类本质也是⼀种封装,当A类跟B类紧密关联,A类实现出来主要就是给B类使⽤,那么可以考虑把A类设计为B的内部类,如果放private/protected位置,那么A类就是B类的专属内部类,其他地⽅都⽤不了。

那么前面的那道题就可以这么写:

class Solution
{
public:
    int Sum_Solution(int n)
    {
        Sum a[n];
        return k;
    }
    static int num;
    static int k;

private:
    class Sum
    {
    public:
        Sum()
        {
            num++;
            k += num;
        }
        static int GetK()
        {
            return k;
        }
   
    };
};
int Solution::num = 0;
int Solution::k = 0;

六、匿名对象

⽤类型(实参)定义出来的对象叫做匿名对象,相⽐之前我们定义的类型对象名(实参)定义出来的叫有名对象。

匿名对象⽣命周期只在当前⼀⾏,⼀般临时定义⼀个对象当前⽤⼀下即可,就可以定义匿名对象。

使用匿名对象的优点在于它可以帮助减少代码的冗余和复杂性,特别是在对象只是用一次的情况下。由于没有持久的引用,匿名对象在使用后会被自动清理,有助于节省内存资源。

#include <iostream>
using namespace std;

class A
{
public:
	A(int v = 1)
		:val(v)
	{}
	~A()
	{
		cout << "~A()->"  << val << endl;
	}
	void print()
	{
		cout << val << endl;
	}
private:
	int val;
};

int main()
{
	A a;
	// 不能这么定义对象,因为编译器无法识别下面是⼀个函数声明,还是对象定义
	//warning C4930: “A aa1(void)”: 未调用原型函数(是否是有意用变量定义的?)
	//A aa1();
	// 但是我们可以这么定义匿名对象,匿名对象的特点不用取名字,
	// 但是他的生命周期只有这一行,我们可以看到下一行他就会自动调用析构函
	A(2);
	A(3);
	A aa(4);
	//无需命名,直接使用类的函数,非常方便
	A().print();
	return 0;
}

在这里插入图片描述

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

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

相关文章

16_网络IPC2-寻址

进程标识 字节序 采用大小模式对数据进行存放的主要区别在于在存放的字节顺序&#xff0c;大端方式将高位存放在低地址&#xff0c;小端方式将高位存放在高地址。 采用大端方式进行数据存放符合人类的正常思维&#xff0c;而采用小端方式进行数据存放利于计算机处理。到目前…

QT使用QPainter绘制多边形维度图

多边形统计维度图是一种用于展示多个维度的数据的图表。它通过将各个维度表示为图表中的多边形的边&#xff0c;根据数据的大小和比例来确定各个维度的长度。 一、简述 本示例实现六边形战力统计维度图&#xff0c;一种将六个维度的战力统计以六边形图形展示的方法。六个维度是…

WebAssembly与JavaScript的交互(1)

前一阵子利用Balazor开发了一个NuGet站点&#xff0c;对WebAssembly进行了初步的了解&#xff0c;觉得挺有意思。在接下来的一系列文章中&#xff0c;我们将通过实例演示的方式介绍WebAssembly的一些基本概念和编程模式。首先我们先来说说什么是WebAssembly&#xff0c;它主要帮…

微调 Florence-2 - 微软的尖端视觉语言模型

Florence-2 是微软于 2024 年 6 月发布的一个基础视觉语言模型。该模型极具吸引力&#xff0c;因为它尺寸很小 (0.2B 及 0.7B) 且在各种计算机视觉和视觉语言任务上表现出色。 Florence 开箱即用支持多种类型的任务&#xff0c;包括: 看图说话、目标检测、OCR 等等。虽然覆盖面…

LRC软件、Adobe Lightroom Classic软件多版本下载+LRC教程

简介&#xff1a; Adobe Lightroom Classic&#xff08;简称LR&#xff09;是Adobe Creative Cloud大家庭中的一款专业的图片管理和编辑工具&#xff0c;用于专业摄影师、摄影爱好者以及所有不断优化数码影像的人等。其目标是以丰富的功能提供高效、一致的体验&#xff0c;帮助…

php基础: 三角形

包含&#xff1a;左三角、左上三角、右三角、右上三角、等腰三角、倒等腰三角。注意空格的数量&#xff0c;因为*号后面加了空格 /*** * 左三角形* param $n* return void*/ function triangleLeft($n){echo <pre>;for ($i 1; $i < $n; $i) {for ($j 1; $j < $i…

对服务器进行基本了解(二)

目录 一. 云服务器数据库 1.查看MYSQL版本 2.查看mysql的运行状态 3.运行mysql 4. 进入mysql的用户 5. 更改用户密码 6. 查找mysql端口号 7. 创建一个数据库 8. 查看用户 9. 查看数据库 10. 显示数据库的表 11. 修改用户的host 12. 对用户赋权 13. 开放指定端…

java.lang.IllegalArgumentException: Illegal character in path at index 40解决方案

大家好,我是爱编程的喵喵。双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。现为CSDN博客专家、人工智能领域优质创作者。喜欢通过博客创作的方式对所学的…

数据库内核研发学习之路(三)创建postgres内置函数

本章之前已经讲明白了我们的postgres如何进行编译安装&#xff0c;这是很重要的一步&#xff0c;接下来就是学会对postgres进行小的改动&#xff0c;然后保证依然能够顺利编译安装运行&#xff01; 本章续讲内容如何创建一个内置函数。 1、内置函数和用户自定义函数的区别 熟…

基于Python+Flask+SQLite的豆瓣电影可视化系统

FlaskMySQLEcharts 基于PythonFlaskSQLite的豆瓣电影可视化系统 Echarts 不支持登录注册&#xff0c;并且信息存储在数据库中 不含爬虫代码&#xff0c;或爬虫代码已失效 简介 基于PythonFlaskMySQL的豆瓣电影可视化系统&#xff0c;采用Echart构建图表&#xff0c;支持自定…

【数据结构】二叉树全攻略,从实现到应用详解

​ &#x1f48e;所属专栏&#xff1a;数据结构与算法学习 &#x1f48e; 欢迎大家互三&#xff1a;2的n次方_ ​ &#x1f341;1. 树形结构的介绍 树是一种非线性的数据结构&#xff0c;它是由n&#xff08;n>0&#xff09;个有限结点组成一个具有层次关系的集合。把它叫做…

动手学深度学习6.3 填充和步幅-笔记练习(PyTorch)

以下内容为结合李沐老师的课程和教材补充的学习笔记&#xff0c;以及对课后练习的一些思考&#xff0c;自留回顾&#xff0c;也供同学之人交流参考。 本节课程地址&#xff1a;填充和步幅_哔哩哔哩_bilibili 代码实现_哔哩哔哩_bilibili 本节教材地址&#xff1a;6.3. 填充和…

旗晟巡检机器人的应用场景有哪些?

巡检机器人作为现代科技的杰出成果&#xff0c;已广泛应用于各个关键场景。从危险的工业现场到至关重要的基础设施&#xff0c;它们的身影无处不在。它们以精准、高效、不知疲倦的特性&#xff0c;担当起保障生产、守护安全的重任&#xff0c;为行业发展注入新的活力。那么&…

【学习笔记】无人机系统(UAS)的连接、识别和跟踪(一)-3GPP TS 23.256 技术规范概述

3GPP TS 23.256 技术规范&#xff0c;主要定义了3GPP系统对无人机&#xff08;UAV&#xff09;的连接性、身份识别、跟踪及A2X&#xff08;Aircraft-to-Everything&#xff09;服务的支持。 3GPP TS 23.256 技术规范&#xff1a; 以下是文档的核心内容总结&#xff1a; UAV系…

深入理解PostgreSql域类型(Domain),灵活定义数据约束,让表结构设计更加严谨

在PostgreSQL中&#xff0c;域&#xff08;Domain&#xff09;是一种用户定义的数据类型&#xff0c;它基于系统内已存在的数据类型&#xff0c;并可以附加约束条件。使用域可以增强数据的完整性和一致性&#xff0c;因为它允许开发者对特定列设定更为具体的规则&#xff0c;比…

Mysql缓存调优的基本知识(附Demo)

目录 前言1. 配置2. 缓存3. 策略 前言 基本的知识推荐阅读&#xff1a; java框架 零基础从入门到精通的学习路线 附开源项目面经等&#xff08;超全&#xff09;Mysql优化高级篇&#xff08;全&#xff09;Mysql底层原理详细剖析常见面试题&#xff08;全&#xff09; MySQL…

集群架构-web服务器(接入负载均衡+数据库+会话保持redis)--15454核心配置详解

紧接着前面的集群架构深化—中小型公司&#xff08;拓展到大型公司业务&#xff09;–下面图简单回顾一下之前做的及故障核心知识总结&#xff08;等后期完全整理后&#xff0c;上传资源希望能帮大家&#xff09; web集群架构-接入负载均衡部署web02服务器等 web集群-搭建web0…

Abaqus基于CT断层扫描的三维重建插件CT2Model 3D

插件介绍 AbyssFish CT2Model 3D V1.0 插件可将采用X射线等方法获取的计算机断层扫描&#xff08;CT&#xff09;图像在Abaqus有限元软件内进行三维重建&#xff0c;进而高效获取可供模拟分析的有限元模型。插件可用于医学影像三维重构、混凝土细观三维重建、岩心数字化等领域…

数据结构-C语言-排序(2)

代码位置&#xff1a;test-c-2024: 对C语言习题代码的练习 (gitee.com) 一、前言&#xff1a; 1.1-排序定义&#xff1a; 排序就是将一组杂乱无章的数据按照一定的规律&#xff08;升序或降序&#xff09;组织起来。(注&#xff1a;我们这里的排序采用的都为升序) 1.2-排序分…

2-36 基于matlab的流行学习算法程序

基于matlab的流行学习算法程序。通过GUI的形式将MDS、PCA、ISOMAP、LLE、Hessian LLE、Laplacian、Dissusion MAP、LTSA八种算法。程序以可视化界面进行展示&#xff0c;可直接调用进行分析。多种案例举例说明八种方法优劣&#xff0c;并且可设置自己数据进行分析。程序已调通&…