类和对象(下+)_const成员、初始化列表、友元、匿名对象

类和对象(下+)


文章目录

  • 类和对象(下+)
  • 前言
  • 一、const成员
  • 二、友元
    • 1.友元函数
    • 2.友元类
  • 三、初始化列表
  • 四、explicit关键字
  • 五、匿名对象
  • 总结


前言

static成员、内部类、const成员、初始化列表、友元、匿名对象


一、const成员

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

其特性如下:

#define  _CRT_SECURE_NO_WARNINGS 1
#include <iostream>
using namespace std;
class Date
{
public:Date(int year, int month, int day){_year = year;_month = month;_day = day;}void Print()//Date* this{cout << _year << "年" << _month << "月" << _day << "日" << endl;}void Print()const //const Date* this{cout << _year << "年" << _month << "月" << _day << "日" << endl;}
private:int _year; // 年int _month; // 月int _day; // 日
};
int main()
{Date d1(2028, 1, 11);d1.Print();const Date d2(2028, 8, 18);d2.Print();return 0;
}

匹配原则(找最合适的)【权限不能放大】:
d1调用第一个Print(带const修饰)
d2调用第二个Print(不带const修饰
在这里插入图片描述
认为是2钟不同的类型,构成函数重载
但是在这里这样使用const是没有意义的。


写一个简单的顺序表

#define  _CRT_SECURE_NO_WARNINGS 1
#include <iostream>
#include <cassert>
using namespace std;class SeqList
{
public:void PushBack(int x){_a[_size++] = x;}size_t size()const{return _size;}int operator[](size_t i){assert(i < _size);return _a[i];}
private:int* _a = (int*)malloc(sizeof(int) * 10);size_t _size = 0;size_t _capacity = 0;
};int main()
{SeqList sl;sl.PushBack(1);sl.PushBack(2);sl.PushBack(3);for (size_t i = 0; i < sl.size(); i++){cout << sl[i]<<" ";//cout << sl.operator[](i) << " ";//与上式等价}return 0;
}

在这里插入图片描述
在这里插入图片描述
但是当我们需要修改sl里的内容时是不可以的,原因是重载operator[ ]返回的是int类型的_a[i]的拷贝,具有常性。
这里需要提一下:在这里插入图片描述
在这里插入图片描述
此时就可以进行修改了。


此时如果又需要一个专门用于打印的函数Print,并且在传参时防止sl对象被修改因此加以const修饰,但是此时又会出现新的报错,如下

#define  _CRT_SECURE_NO_WARNINGS 1
#include <iostream>
#include <cassert>
using namespace std;class SeqList
{
public:void PushBack(int x){_a[_size++] = x;}size_t size(){return _size;}int& operator[](size_t i){assert(i < _size);return _a[i];}
private:int* _a = (int*)malloc(sizeof(int) * 10);size_t _size = 0;size_t _capacity = 0;
};void Print(const SeqList& sl)
{for (size_t i = 0; i < s.size(); i++){cout << sl[i] << " ";}
}int main()
{SeqList sl;sl.PushBack(1);sl.PushBack(2);sl.PushBack(3);Print(sl);return 0;
}

在这里插入图片描述

报错原因出在:const对象调用非const对象,存在权限放大的问题
解决方法:

	size_t size() const{return _size;}//只读int& operator[](size_t i) const{assert(i < _size);return _a[i];}//读 or 写都可以  		//与上一个代码块构成函数重载int& operator[](size_t i){assert(i < _size);return _a[i];}

只需要在size()和[ ]重载函数中加以const修饰this指针,即可。

同样看下面这段日期类

class Date
{
public:Date(int year, int month, int day){_year = year;_month = month;_day = day;}void Print()const{cout << _year << "年" << _month << "月" << _day << "日" << endl;}bool operator<(const Date& d)const{//this->Print();//this 是非const 的,可以调用const 的Printif (_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;}return false;}
private:int _year; // 年int _month; // 月int _day; // 日
};int main()
{const Date d1(2028, 1, 11);Date d2(2028, 8, 18);cout << (d1 < d2) << endl;cout << (d1.operator<(d2)) << endl;//与上行代码等价return 0;
}

const Date d1(2028, 1, 11);当d1加上const时,如果operator<不加const的话,会出现错误,原因依旧是d1的类型是const Date*类型,(属于权限放大)


二、友元

1.友元函数

友元函数的几点特性:

  • 友元函数可访问类的私有和保护成员,但不是类的成员函数(没有this指针)
  • 友元函数不能用const修饰
  • 友元函数可以在类定义的任何地方声明,不受访问限定符限制
  • 一个函数可以是多个类的友元函数
  • 友元函数的调用和普通函数的调用原理相同

还是之前说的日期类,此时我们想要重载一下"<<"如下

	void operator<<(ostream& out){out << _year << "年" << _month << "月" << _day << "日" << endl;}

但是当我们想要使用重载的"<<“时,会出现以下情况:
在这里插入图片描述
在这里插入图片描述
因此为了能够正常使用,我们在全局定义关于”<<"的重载函数,但在全局定义又会出现无法访问成员变量的问题,因此此时就需要在Date中进行友元声明,这样在全局定义的函数就可以访问类成员变量了

friend void operator<<(ostream& out, const Date& d);

接下来为了满足cout<<d1<<d2;需要以上返回out,即:

	ostream& operator<<(ostream& out, const Date& d){out << d._year << "年" << d._month << "月" << d._day << "日" << endl;return out;}

在这里插入图片描述

2.友元类

友元类的所有成员函数都可以是另一个类的友元函数,都可以访问另一个类中的非公有成员

  1. 友元关系是单向的,不具有交换性
  2. 友元关系不能传递
  3. 友元关系不能继承
class Time
{friend class Date1;
public:friend class Date;	// 声明日期类为时间类的友元类,//则在日期类中就直接访问Time类中的私有成员变量Time(int hour = 0, int minute = 0, int second = 0): _hour(hour), _minute(minute), _second(second){}
private:int _hour; int _minute; int _second;
};class Date1
{
public:Date1(int year, int month, int day){_year = year;_month = month;_day = day;}void SetTimeOfDate(int hour, int minute, int second){// 直接访问时间类私有的成员变量_t._hour = hour;_t._minute = minute;_t._second = second;}void Print()const{cout << _year << "年" << _month << "月" << _day << "日" << endl;}
private:int _year; // 年int _month; // 月int _day; // 日Time _t;
};

三、初始化列表

就像这样

public://初始化列表,是每个成员定义的地方Date(int year = 1900, int month = 1, int day = 1): _year(year), _month(month), _day(day){}
private://每个成员的声明int _year;int _month;int _day;

需要注意的几点是:

  1. 每个成员变量在初始化列表中只能出现一次(初始化只能初始化一次)
  2. 类中包含以下成员,必须放在初始化列表位置进行初始化:
    • 引用成员变量
    • const成员变量
    • 自定义类型成员(且该类没有默认构造函数时)
  3. 尽量使用初始化列表初始化,因为不管你是否使用初始化列表,对于自定义类型成员变量,一定会先使用初始化列表初始化。
  4. 成员变量在类中声明次序就是其在初始化列表中的初始化顺序与其在初始化列表中的先后次序无关
    在这里插入图片描述
    在构造函数的初始化列表阶段,对内置类型用随机值进行初始化,对自定义类型会调用它的默认构造
    在这里插入图片描述

当_day给了缺省值 int _day=1;
在这里插入图片描述
_day给了缺省值,把它初始化成1 了;接下来还要走函数体变为18,由此可知()缺省值就是给初始化列表用的!!

但是初始化列表也有不能解决初始化问题
(比如要求数组_array初始化一下(在函数体中中memset初始化))

四、explicit关键字

class A
{
public:A(int i):_a(i){cout << "A" << endl;}
private:int _a;
};int main()
{A aa1(1);A aa2 = 2;
}

在这里插入图片描述
因为

  • 单参数构造函数的隐式类型转换
  • 用2调用A构造函数生成一个临时对象,再用这个对象去拷贝构造aa2
  • 编译器会再优化,优化用2直接构造
    在这里插入图片描述
    >A& ref = 2;
    存在类型转换,会生成临时对象,2具有常量性,ref不能引用临时对象,也就是存在权限放大问题,加上const就可以了即:

const A& ref = 3;

紧接着还有一个问题,那么为什么这里ret可以引用2,因为因为这个单参数的函数支持隐式类型转换,单参数函数这个参数(2)的的整型值能转换成一个A的对象,就可以引用了。

如果说bu想让隐式类型转换发生,可以加关键字explicit,加在构造函数的位置

	explicit  A(int i):_a(i){cout << "A" << endl;}

此时在这里插入图片描述
在这里插入图片描述

以上是讨论的单参数的,那么如果是多参数的呢?

c++11支持多参数的转换

class B
{
public:B(int b1,int b2):_b1(b1),_b2(b2){cout << "B" << endl;}private:int _b1;int _b2;
};
int main()
{B bb1(1, 1);B bb2 = {2,2};const B& ref = { 3,3 };//同样的,不加const就不支持引用return 0;
}

道理同上,如果不想让隐式类型转换发生,使用关键字explicit

五、匿名对象

首先来比较一下有名对象和匿名对象:

  • 有名对象 特点:生命周期在当前局部域

A aa4(4);

  • 匿名对象 特点:生命周期只在这一行

A (5);

匿名对象可以用来传参,不用先创建变量,再传参

class A
{
public:explicit  A(int i):_a(i){cout << "A" << endl;}
private:int _a;
};class SeqList
{
public:void PushBack(A x){_a[_size++] = x;}size_t size() const{return _size;}A& operator[](size_t i) const{assert(i < _size);return _a[i];}private:A* _a = (A*)malloc(sizeof(A) * 10);size_t _size = 0;size_t _capacity = 0;
};int main()
{SeqList s;A aa3(3);s.PushBack(aa3);s.PushBack(A(4));//利用匿名对象,直接传参return 0;
}

当在一些特定场景下,适当使用匿名对象可以起到简化代码的作用。

总结

😁 ;

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

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

相关文章

如何使用共享GPU平台搭建LLAMA3环境(LLaMA-Factory)

0. 简介 最近受到优刻得的使用邀请&#xff0c;正好解决了我在大模型和自动驾驶行业对GPU的使用需求。UCloud云计算旗下的[Compshare](https://www.compshare.cn/? ytagGPU_lovelyyoshino_Lcsdn_csdn_display)的GPU算力云平台。他们提供高性价比的4090 GPU&#xff0c;按时收…

【安装笔记-20240608-Linux-动态域名更新服务之YDNS】

安装笔记-系列文章目录 安装笔记-20240608-Linux-动态域名更新服务之YDNS 文章目录 安装笔记-系列文章目录安装笔记-20240608-Linux-动态域名更新服务之YDNS 前言一、软件介绍名称&#xff1a;YDNS主页官方介绍 二、安装步骤测试版本&#xff1a;openwrt-23.05.3-x86-64注册填…

1-5 C语言操作符

C语言提供了非常丰富的操作符&#xff0c;使得C语言使用起来非常的方便 算数操作符&#xff1a; 加 减 乘 除 取模 【 - * / %】 注&#xff1a;除号的两端都是整数的时候执行的是整数的除法&#xff0c;如果…

65. UE5 RPG 实现远程攻击

前面&#xff0c;我们实现了敌人的近战普通攻击&#xff0c;还兼容了对于没有武器的敌人的攻击。有近战就要有远程&#xff0c;这一篇&#xff0c;我们实现一下敌人的远程攻击。 首先&#xff0c;由于创建的资源增多&#xff0c;我们将GA和GE按敌人和英雄的类别分开&#xff0c…

MySQL—多表查询—联合查询

一、引言 之前学习了连接查询。现在学习联合查询。 union&#xff1a;联合、联盟 对于union查询&#xff0c;就是把多次查询的结果合并起来&#xff0c;形成一个新的查询结果集 涉及到两个关键字&#xff1a;union 和 union all 注意&#xff1a; union 会把上面两个SQL查询…

C++的STL 中 set.map multiset.multimap 学习使用详细讲解(含配套OJ题练习使用详细解答)

目录 一、set 1.set的介绍 2.set的使用 2.1 set的模板参数列表 2.2 set的构造 2.3 set的迭代器 2.4 set的容量 2.5 set的修改操作 2.6 set的使用举例 二、map 1.map的介绍 2.map的使用 2.1 map的模板参数说明 2.2 map的构造 2.3 map的迭代器 2.4 map的容量与元…

深圳中赢娱乐控股集团至江西省宜春市袁州区访问交流

2024年6月7日&#xff0c;深圳中赢娱乐控股集团受邀来到江西省宜春市袁州区就“短剧文旅”项目展开深度座谈&#xff0c;并与飞剑潭乡达成合作意向。 下午2:30&#xff0c;深圳中赢控股集团董事李平进带团队一行12人&#xff0c;访问宜春市袁州区&#xff0c;宜春市副市长谢萍、…

嵌入式学习——4——C++中的动态内存分配和回收(堆区)

1、内存的分配与回收 C语言中使用的是malloc和free函数进行动态内存分配和回收的。 C中依然可以使用上述的两个函数来完成动态内存分配和回收的。 C也给用户提供了两个关键字new、delete来完成动态内存分配和回收的 单个分配、回收 //在堆区申请了int类型的大小空间&#xff0c…

html--圣诞树

将以下代码保存到txt文件中&#xff0c;并改名为xx.html <html> <head> <title>圣诞树</title> <meta charset"utf-8" > <style> html, body { width: 100%; height: 100%; margin: 0; padding: 0; border: 0; } div { margin: …

Java--命令行传参

1.有时你希望运行一个程序时再传递给它消息&#xff0c;这要靠传递命令行参数给main&#xff08;&#xff09;函数实现 2.选中文件右键找到如图选项并打开 3.在文件地址下输入cmd空格符号&#xff0c;再按回车调出命令窗口 4.如图一步步进行编译&#xff0c;在向其传入参数&…

echarts柱状图的背景动态效果

这里的动态效果实现原理&#xff0c;就是相当于柱状图多了一组同系列数据&#xff0c;其值与数组展示数据一致&#xff08;类似下图&#xff09; 即&#xff0c;柱形图的每一个柱体都有它对应的另外一个柱体 其中白色柱体要与展示柱体重合&#xff0c;效果类似与给柱体做背景…

IEDA 默认依赖概述

IEDA 默认依赖概述 目录概述需求&#xff1a; 设计思路实现思路分析1.AI Azure OpenAlAzure Al SearchAmazon BedrockChroma Vector DatabaseMilvus Vector DatabaseMistral AlNeo4J Vector DatabaseOllamaOpenAlPGvector Vector DatabasePinecone Vector DatabasePostgresMLRe…

Java抽象队列同步器AQS

AQS介绍 AQS是一个抽象类&#xff0c;主要用来构建锁和同步器。 public abstract class AbstractQueuedSynchronizer extends AbstractOwnableSynchronizer implements java.io.Serializable { }AQS为构建锁和同步器提供了一些通用功能的实现&#xff0c;因此&#xff0c;使用…

关于音乐播放器与系统功能联动功能梳理

主要实现功能&#xff1a; 一、通知栏播放显示和控制 二、系统下拉栏中播放模块显示同步 三、与其他播放器状态同步&#xff1a;本应用播放时暂停其他应用播放&#xff0c;进入其他应用播放时&#xff0c;暂停本应用的后台播放 通知栏播放的显示和控制&#xff1a; 通过Not…

数字人实战第一天——最新数字人MuseTalk效果展示

最新数字人MuseTalk效果展示 MuseTalk 是由腾讯团队开发的先进技术&#xff0c;项目地址&#xff1a;GitHub - TMElyralab/MuseTalk: MuseTalk: Real-Time High Quality Lip Synchorization with Latent Space Inpainting&#xff0c;它是一个实时的音频驱动唇部同步模型。该模…

简单通用的系统安装、备份、还原方法,支持 ARM 系统【Ventory+FirePE+DiskGenius】

文章目录 0. 简介1. 制作 Ventory 启动盘1.1. 下载 Ventory1.2. 制作 Ventory 启动盘 2. 添加 FirePE 等系统镜像到启动盘2.1. 下载 FirePE2.2. 导出 .iso 系统镜像文件2.3. .iso 系统镜像文件添加至启动盘 3. 启动 FirePE 等系统镜像3.1. 在 bios 中选择启动盘启动3.2. 启动系…

访问网站时IP被阻止?原因及解决方法

在互联网上&#xff0c;用户可能会面临一个令人困扰的问题——当尝试访问某个特定的网站时&#xff0c;却发现自己的IP地址被该网站屏蔽。 IP地址被网站屏蔽是一个相对常见的现象&#xff0c;而导致这种情况的原因多种多样&#xff0c;包括恶意行为、违规访问等。本文将解释IP地…

大学搜题软件网课?推荐五个搜题软件和学习工具 #其他#经验分享#知识分享

大学生活中&#xff0c;选择适合自己的学习工具能够提高学习效率&#xff0c;让学习更加轻松愉快。 1.彩虹搜题 这个是公众号 提供了各大教材以及网课平台的练习题答案&#xff0c;强大的平台支持&#xff0c;无论是智慧树还是MOOC&#xff0c;只有老师们用不到&#xff0c;…

理解JVM内存模型与Java内存模型(JMM)

理解JVM内存模型与Java内存模型&#xff08;JMM&#xff09; 在Java程序的运行过程中&#xff0c;内存管理和线程的同步是两个重要的概念。本文将深入探讨JVM内存模型&#xff08;Java Virtual Machine Memory Model&#xff09;和JMM&#xff08;Java Memory Model&#xff0…

实现开源可商用的 ChatPDF RAG:密集向量检索(R)+上下文学习(AG)

实现 ChatPDF & RAG&#xff1a;密集向量检索&#xff08;R&#xff09;上下文学习&#xff08;AG&#xff09; RAG 是啥&#xff1f;实现 ChatPDF怎么优化 RAG&#xff1f; RAG 是啥&#xff1f; RAG 是检索增强生成的缩写&#xff0c;是一种结合了信息检索技术与语言生成…