运算符重载(上)

目录

  • 运算符重载
    • 日期类的比较
      • 判断日期是否相等
      • 判断日期大小
  • 赋值运算符重载
    • 赋值运算符重载格式
    • 赋值运算符只能重载成类的成员函数不能重载成全局函数
    • 用户没有显式实现时,编译器会生成一个默认赋值运算符重载,以值的方式逐字节拷贝

感谢各位大佬对我的支持,如果我的文章对你有用,欢迎点击以下链接
🐒🐒🐒 个人主页
🥸🥸🥸 C语言
🐿️🐿️🐿️ C语言例题
🐣🐣🐣 python
🐓🐓🐓 数据结构C语言
🐔🐔🐔 C++
🐿️🐿️🐿️ 文章链接目录

运算符重载

C++为了增强代码的可读性引入了运算符重载
运算符重载是具有特殊函数名的函数,也具有其返回值类型,函数名字以及参数列表,其返回值类型与参数列表与普通的函数类似。
函数名字为:关键字operator后面接需要重载的运算符符号。
函数原型:返回值类型 operator操作符(参数列表)
注意:
不能通过连接其他符号来创建新的操作符(可以连接C/C++语法中存在的操作符):比如operator@
重载操作符必须有一个类类型参数,不能去重载运算符改变内置类型的行为
用于内置类型的运算符,其含义不能改变,例如:内置的整型+,不能改变其含义
作为类成员函数重载时,其形参看起来比操作数数目少1,因为成员函数的第一个参数为隐藏的this
注意以下5个运算符不能重载(这个经常在笔试选择题中出现)

.*   ::   sizeof   ?:   . 

日期类的比较

class Date
{
public:
	Date(int year = 1900, int month = 1, int day = 1)
	{
		_year = year;
		_month = month;
		_day = day;
	}
	//private:
	int _year;
	int _month;
	int _day;
};
int main()
{
	Date d1(2024,5,10);
	Date d2(2024,5,11);
	return 0;
}

日期类是不能直接比较的,比如d1<d2,因为日期类是一个自定义类型,他里面的内置类型是不确定的,可能里面有int char double…,编译器没有一个具体的比较方法,所以具体的比较方法要我们自己去规定

一般的写法是自己创造一个函数

判断日期是否相等

bool DateEquel(const Date& x, const Date& y)
{
	return x._year == y._year
		&& x._month == y._month
		&& x._day == y._day;
}

在这里插入图片描述

判断日期大小

bool DateLess(const Date& x, const Date& y)
{
	if (x._year < y._year)
	{
		return true;
	}
	else if (x._year == y._year)
	{
		if (x._month < y._month)
			return true;
		else if (x._month == y._month)
		{
			return x._day < y. _day;
		}
	}
	return false;
}

在这里插入图片描述

对应函数来说,取一个容易理解的函数名是非常重要的,如果将上面的函数全部取成func1 func2…这样的名字,别人用起来会非常的难受,所以C++中采用运算符重载来提高可读性

采用运算符重载前:
1:bool DateEquel(const Date& x, const Date& y)
2:bool DateLess(const Date& x, const Date& y)
采用运算符重载后
1:bool operator==(const Date& x, const Date& y)
2:bool operator<(const Date& x, const Date& y)

在这里插入图片描述
可以看到运算符重载只是改变了函数名,函数内容的实现是没有变化的

事实上在用了运算符重载实现函数后,我们可以直接就进行比较,比如d1==d2,d1<d2
在这里插入图片描述
这里的红波浪是因为优先级的问题,<<符号的优先级比= =和<高,所以需要加一个括号,这样在用函数比较的时候就非常方便了
在这里插入图片描述
为什么有了运算符重载后可以直接就进行比较呢?
因为这个和this指针非常相似,这里的比较是隐示调用了对应的函数
比如d1<d2其实是d1.operator<(d2)

运算符重载可以简化非常多的函数,但是像减 和 除就需要注意顺序问题

运算符重载函数访问对象的成员会受到限制,之前可以正常访问是将private给屏蔽掉了,所以没有报错,当取消屏蔽后,就变成了不可访问

在这里插入图片描述
一般的解决方法是可以在私有的前面加上函数GetYear…,让这些函数返回你需要的成员

class Date
{
public:
	Date(int year = 1900, int month = 1, int day = 1)
	{
		_year = year;
		_month = month;
		_day = day;
	}
	int GetYear()
	{
		return _year;
	}
	int GetDay()
	{
		return _month;
	}
	int GetDay()
	{
		return _day;
	}
	private:
	int _year;
	int _month;
	int _day;
};

另一种方式就是在类里面访问成员变量

class Date
{
public:
	Date(int year = 1900, int month = 1, int day = 1)
	{
		_year = year;
		_month = month;
		_day = day;
	}
	bool operator==(const Date& x, const Date& y)
	{
		return x._year == y._year
			&& x._month == y._month
			&& x._day == y._day;
	}
	bool operator<(const Date& x, const Date& y)
	{
		if (x._year < y._year)
		{
			return true;
		}
		else if (x._year == y._year)
		{
			if (x._month < y._month)
				return true;
			else if (x._month == y._month)
			{
				return x._day < y._day;
			}
		}
		return false;
	}
	private:
	int _year;
	int _month;
	int _day;
};

这样写后会报错,说参数太多,其实是因为有隐含的this指针导致的,也就是说有一个参数已经传递过去了,不需要我们再传一次
在这里插入图片描述
只需要稍微改一下就可以了

class Date
{
public:
	Date(int year = 1900, int month = 1, int day = 1)
	{
		_year = year;
		_month = month;
		_day = day;
	}
	bool operator==( const Date& y)
	{
		return _year == y._year
			&& _month == y._month
			&& _day == y._day;
	}
	bool operator<( const Date& y)
	{
		if (_year < y._year)
		{
			return true;
		}
		else if (_year == y._year)
		{
			if (_month < y._month)
				return true;
			else if (_month == y._month)
			{
				return _day < y._day;
			}
		}
		return false;
	}
	private:
	int _year;
	int _month;
	int _day;
};

在这里插入图片描述

赋值运算符重载

赋值运算符重载是针对两个已经存在的对象,其中一个拷贝赋值给另一个

赋值运算符重载格式

参数类型:const T&,传递引用可以提高传参效率
返回值类型:T&,返回引用可以提高返回的效率,有返回值目的是为了支持连续赋值,并检测是否自己给自己赋值
返回*this :要符合连续赋值的含义

class Date
{
public:
	Date(int year, int month, int day)
	{
		_year = year;
		_month = month;
		_day = day;
	}
		void operator=(const Date & d)
		{
			_year = d._year;
			_month = d._month;
			_day = d._day;
		}
		void Print()
		{
			cout << _year << "/" << _month << "/" << _day << endl;
		}
	private:
		int _year;
		int _month;
		int _day;
};
	int main()
	{
		Date d1(2024, 5, 11);
		Date d2(2024, 5, 10);
		d2 = d1;
		d1.Print();
		d2.Print();

	}

赋值运算符重载其实和拷贝构造差不多,只是通过operator使我们在用的时候更加方便
在这里插入图片描述
在内置类型中可以支持连续赋值,比如i=j=k,这里的赋值顺序是先让k赋值给j,然后j再给i赋值,最后左操作数i作为返回值,这个返回值是可以验证的
在这里插入图片描述

运算符重载想要实现也是可以的,只需要让他返回每次赋值的值就可以了,那如何返回每次的赋值呢?
我们知道类中有多个成员变量,就像结构体一样,要想得到结构体里面的成员变量,我们需要获得结构体的指针,然后通过这个指针去访问结构体,类也是一样的,那怎么得到类的指针呢?
this指针可以解决这个问题
但是这样写后却出来了一个问题
在这里插入图片描述
这是因为我们的返回类型是Date,而this指针是一个指针,所以出现了错误
需要注意的是我们真正想要返回的是一个类,因为连续赋值是类与类直接进行赋值,如果一个类赋值完后返回一个指针,然后用这个指针给下一个类进行赋值,这显然不合理,所以我们需要对this指针进行解引用,让返回的结果是一个类

	Date operator=(const Date & d)
	{
		_year = d._year;
		_month = d._month;
		_day = d._day;
		return *this;
	}

传值返回有一个缺点就是返回的值并不是当前的对象,而是他的拷贝
比如d1=d2=d3,这种连续赋值的情况由于传值返回是拷贝,所以会调用拷贝函数,而调用拷贝构造函数会建立栈帧,这样的话有点浪费空间,所以可以将传值返回变成传引用返回

	Date operator=(const Date & d)
	{
		_year = d._year;
		_month = d._month;
		_day = d._day;
		return *this;
	}

上面的代码是可以支持自己给自己赋值的,比如d1=d1,为了不让这种情况出现,需要加一个条件判断this!=&d,注意&d这里表示的是取地址,不是引用

	Date operator=(const Date & d)
	{
		if(this!=&d)
		{	_year = d._year;
			_month = d._month;
			_day = d._day;
		}
		return *this;
	}

赋值运算符只能重载成类的成员函数不能重载成全局函数

原因:赋值运算符如果不显式实现,编译器会生成一个默认的。

此时用户再在类外自己实现一个全局的赋值运算符重载,就和编译器在类中生成的默认赋值运算符重载冲突了,所以赋值运算符重载只能是类的成员函数。

用户没有显式实现时,编译器会生成一个默认赋值运算符重载,以值的方式逐字节拷贝

注意:内置类型成员变量是直接赋值的,而自定义类型成员变量需要调用对应类的赋值运算符重载完成赋值。

既然编译器生成的默认赋值运算符重载函数已经可以完成字节序的值拷贝了,还需要自己实
现吗,这个还是和之前拷贝构造函数涉及到的问题一样,需要分浅拷贝和深拷贝去处理

注意:如果类中未涉及到资源管理,赋值运算符是否实现都可以;一旦涉及到资源管理则必须要实现。

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

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

相关文章

Blender导出fbx模型,导入到ue5中模型丢失纹理材质

UE5系列文章目录 文章目录 UE5系列文章目录前言一、问题原因二、最终效果 前言 Blender导出fbx模型&#xff0c;导入到ue5中&#xff0c;发现模型丢失纹理材质&#xff0c;里面的原神人物模型妮露居然是白模&#xff0c;郁闷了大半天 一、问题原因 我在Blender导出fbx文件时…

如何高效创建与配置工程环境:零基础入门

新书上架~&#x1f447;全国包邮奥~ python实用小工具开发教程http://pythontoolsteach.com/3 欢迎关注我&#x1f446;&#xff0c;收藏下次不迷路┗|&#xff40;O′|┛ 嗷~~ 目录 一、工程环境的搭建与准备 二、配置虚拟环境与选择解释器 三、编写代码与自动添加多行注释 …

Mybatis Cache(一)MybatisCache+Redis

前面提到了&#xff0c;使用mybatis cache&#xff0c;一般是结合redis使用。 一、demo 1、数据表 create table demo.t_address (id int auto_incrementprimary key,address_name varchar(200) null,address_code varchar(20) null,address_type int n…

6.小程序页面布局 - 账单明细

文章目录 1. 6.小程序页面布局 - 账单明细1.1. 竞品1.2. 布局分析1.3. 布局demo1.4. 页面实现-头部1.5. 账单明细1.5.1. 账单明细-竞品分析1.5.2. 账单明细-实现1.5.2.1. 账单明细-实现-mock数据1.5.2.2. 每日收支数据的聚合整理1.5.2.3. 页面scroll-view 1.6. TODO 1. 6.小程序…

雷电预警监控系统:守护安全的重要防线

TH-LD1在自然界中&#xff0c;雷电是一种常见而强大的自然现象。它既有震撼人心的壮观景象&#xff0c;又潜藏着巨大的安全风险。为了有效应对雷电带来的威胁&#xff0c;雷电预警监控系统应运而生&#xff0c;成为现代社会中不可或缺的安全防护工具。 雷电预警监控系统的基本…

Convolutional Occupancy Networks【ECCV2020】

论文&#xff1a;https://arxiv.org/pdf/2003.04618 代码&#xff1a;GitHub - autonomousvision/convolutional_occupancy_networks: [ECCV20] Convolutional Occupancy Networks 图 1&#xff1a;卷积占据网络。传统的隐式模型 (a) 由于其全连接网络结构&#xff0c;表现能力…

政策及需求多因素驱动下 中国适老化改造市场空间大

政策及需求多因素驱动下 中国适老化改造市场空间大 适老化改造是为了提高老年人居住环境的舒适度和安全性&#xff0c;满足老年人居住需求进行的建筑改造&#xff0c;根据住房和城乡建设部城市建设司发布的《城市居家适老化改造指导手册》可以将适老化改造分为基础性改造和提升…

Spring Cloud 系列之Gateway:(9)初识网关

传送门 Spring Cloud Alibaba系列之nacos&#xff1a;(1)安装 Spring Cloud Alibaba系列之nacos&#xff1a;(2)单机模式支持mysql Spring Cloud Alibaba系列之nacos&#xff1a;(3)服务注册发现 Spring Cloud 系列之OpenFeign&#xff1a;(4)集成OpenFeign Spring Cloud …

【小笔记】如何在docker中更新或导入neo4j数据?

如何在docker中更新或导入neo4j数据&#xff1f; &#xff08;1&#xff09;背景&#xff1a; 我尝试了4.4.9和5.19.0版本的Neo4j社区版&#xff0c;基于他们的镜像创建容器后&#xff0c;需要导入我准备好的csv文件或dump文件&#xff0c;因为数据量非常大&#xff0c;所以采…

装备制造项目管理软件:奥博思PowerProject项目管理系统

数字化正逐步改变着制造方式和企业组织模式。某制造企业领导层透露&#xff0c;在采用数字化项目管理模式后&#xff0c;企业的发展韧性更加强劲&#xff0c;构筑起了竞争新优势&#xff0c;企业产品研制周期缩短25%&#xff0c;生产效率提升18%。 随着全球经济的发展&#xf…

北理工提出 LTrack 双摄像头系统 | 专注于暗场景多目标跟踪,自动驾驶和夜间监控的福音!

低光照场景在现实世界应用中很普遍&#xff08;例如自动驾驶和夜间监控&#xff09;。最近&#xff0c;在各种实际用例中的多目标跟踪受到了很多关注&#xff0c;但在暗场景中的多目标跟踪却鲜少被考虑。 在本文中&#xff0c;作者专注于暗场景中的多目标跟踪。为了解决数据集…

【电子学会】2023年09月图形化一级 -- 芝麻开门

芝麻开门 1. 准备工作 &#xff08;1&#xff09;删除小猫角色&#xff0c;添加角色Key&#xff1b; &#xff08;2&#xff09;删除白色背景&#xff0c;添加背景Castle 1和Pathway。 2. 功能实现 &#xff08;1&#xff09;点击绿旗&#xff0c;钥匙在舞台中间&#xff…

Git 的安装和使用

一、Git 的下载和安装 目录 一、Git 的下载和安装 1. git 的下载 2. 安装 二、Git 的基本使用-操作本地仓库 1 初始化仓库 1&#xff09;创建一个空目录 2&#xff09;git init 2 把文件添加到版本库 1&#xff09;创建文件 2&#xff09;git add . 3&#xff09;g…

51单片机简单控制180度舵机

代码&#xff1a; 链接&#xff1a;https://pan.baidu.com/s/1K9dg2NwRhy49db_O_hqv-g?pwd1234 提取码&#xff1a;1234 一、路线 我在了解这个舵机之前最像想看到的是一个完全的路径。 比如我想学习b站上那个智能门锁&#xff0c;那就得每个模块的基本代码都会才能结合各…

​​​【收录 Hello 算法】第 10 章 搜索

目录 第 10 章 搜索 本章内容 第 10 章 搜索 搜索是一场未知的冒险&#xff0c;我们或许需要走遍神秘空间的每个角落&#xff0c;又或许可以快速锁定目标。 在这场寻觅之旅中&#xff0c;每一次探索都可能得到一个未曾料想的答案。 本章内容 10.1 二分查找10.2 二…

智慧展厅设计的难点有哪些

1、运用先进的展示技术 将全息影像、三维投影、虚拟现实、人机互动等技术做做完美衔接&#xff0c;把展厅的内容展示做到丰富多彩&#xff0c;从而让展厅富有科技感和艺术性。 2、内容要生动有趣 从而更好地吸引参观者。展厅设计师要与客户有良好深入的沟通&#xff0c;搜集与整…

struct.unpack_from()学习笔记

struct.unpack_from(fmt,b_data,offset) 按照指定的格式fmt&#xff0c;从偏移位置offset&#xff0c;对b_data开始解包&#xff0c;返回数据格式是一个元组(v1,v2…) fmt可以有&#xff1a; _struct.py: The remaining chars indicate types of args and must match exactly;…

WPF之容器标签之Canvas布局标签

Canvas: 定义一个区域&#xff0c;可在其中使用相对于 Canvas 区域的坐标以显式方式来定位子元素。 实例 可以在子标签使用Canvas属性设置定位 <Canvas Width"500" Height"300"><StackPanel Width"100" Height"100"Backgro…

详解最新版RabbitMQ 基于RPM 方式的安装

如何选择安装版本 已经不支持的发布系列 版本最后补丁版本首次发布时间停止更新时间3.73.7.282017年11月28日2020年09月30日3.63.6.162015年12月22日2018年05月31日3.53.5.82015年03月11日2016年10月31日3.43.4.42014年10月21日2015年10月31日3.33.3.52014年04月02日2015年03…

列表元素添加的艺术:从单一到批量

新书上架~&#x1f447;全国包邮奥~ python实用小工具开发教程http://pythontoolsteach.com/3 欢迎关注我&#x1f446;&#xff0c;收藏下次不迷路┗|&#xff40;O′|┛ 嗷~~ 目录 一、引言 二、向列表中添加单一元素 1. append方法 2. insert方法 三、向列表中添加批量…