C++:类和对象:运算符重载

news2025/7/13 10:09:38

前言:

运算符重载概念:对已有的运算符重新进行定义,赋予其另一种功能,以适应不同的数据类型。

1:加号运算符重载

对于内置的数据类型, 编译器知道如何运算,可以很直观的得到结果

int a = 10;

int b = 10;

int c = a+b;

但是现在我们有个 Person类,类中有两个成员变量m_A和 m_B,现在我们有两个 Person对象(P1和P2) ,如果我们直接通过加号运算符+ 将两个对象变量相加创建出一个新对象,这样的行为肯定是不行的。那么我们应该想个办法:通过自己写个成员函数,实现两个对象的成员变量相加并返回新对象。

class Person {
public:
	int m_A;
	int m_B;

	Person PersonAndPerson(Person& p) {
		Person temp;
		temp.m_A = this->m_A + p.m_A;
		temp.m_B = this->m_B + p.m_B;
		return temp;
	}
};

但是这个方法也有缺点,如果多个人都这样实现函数,那么可能每个人定义的函数名不一样,这样会造成混乱,所以编译器干脆给我们提供一个函数名:oprator+ 

 1.1:成员函数重载+号运算符

class Person {
public:
	Person() {}
	Person(int a, int b) {
		this->m_A = a;
		this->m_B = b;
	}
	int m_A;
	int m_B;

	Person operator+(Person& p) {
		Person temp;
		temp.m_A = this->m_A + p.m_A;
		temp.m_B = this->m_B + p.m_B;
		return temp;
	}
};

int main() {
	Person P1(10, 10);
	Person P2(20, 20);
	// 本质是:Person p3 = p1.operator+(p2),重载加号运算符
	Person P3 = P1 + P2;
}

 

从运行结果可知:可以直接通过 + 号 完成两个对象的成员变量相加。 

1.2 全局函数重载+号运算符 

我们也可以通过 全局函数 重载+号运算符

运行结果可知:通过 + 符号可以直接完成两个对象的成员相加 

1.3 运算符重载的函数重载

现在我们想让 Person 变量和 int类型变量相加,即将 int类型的变量值加在 Person的两个成员变量上,那么我们可以对运算符重载使用函数重载。

#include<iostream>
using namespace std;
class Person {
public:
	Person() {}
	Person(int a, int b) {
		this->m_A = a;
		this->m_B = b;
	}
	int m_A;
	int m_B;
};

Person operator+(Person& p1,Person& p2) {
	Person temp;
	temp.m_A = p1.m_A + p2.m_A;
	temp.m_B = p1.m_B + p2.m_B;
	return temp;
}

// 运算符重载,可以发生函数重载
Person operator+(const Person& p2, int value) {
	Person temp;
	temp.m_A = p2.m_A + value;
	temp.m_B = p2.m_B + value;
	return temp;
}

int main() {
	Person P1(30, 30);
	// 本质是:Person p3 = operator+(p1,p2),重载加号运算符
	Person P3 = P1 + 10;
	cout << "p3 m_A = " << P3.m_A << " ,p3 m_B = " << P3.m_B << endl;
}

运行结果可知:调用的是下面这个重载运算符

Person operator+(const Person& p2, int value) 

需要注意的是

1:对于内置的数据类型表达式的运算符是不可以改变的

2:请勿滥用运算符重载 

2:左移运算符重载 

全局函数来重载左移运算符,大致框架如下:

#include<iostream>
#include<string>
using namespace std;
class Person {
public:
	Person(int a, int b) {
		this->m_A = a;
		this->m_B = b;
	}
public:
	int m_A;
	int m_B;
};

// 全局函数实现左移重载
// 实现 cout << p
ostream& operator<<(ostream& out,Person& p) {
	out << "m_A :" << p.m_A << "  b:" << p.m_B;
	return out;
}

int main() {
	Person p1(10, 20);
	cout << p1<< endl;
	return 0;
}

运行结果可知:可以正确重载 << 运算符 并输出了 对象 p1的成员变量。 

3:递增运算符重载 

3.1 重载前置

#include<iostream>
using namespace std;
class MyInter {
	friend ostream& operator<<(ostream& cout, MyInter myint);
public:
	MyInter(){
		m_Num= 0;
	}
	MyInter& operator++() {
		m_Num++; // 先进行++运算
		return *this;
	}
private:
	int m_Num;
};

ostream& operator<<(ostream& cout, MyInter myint) {
	cout << myint.m_Num;
	return cout;
}

int main() {
	MyInter myint;
	cout << myint << "  自增一次:";
	cout << ++ myint << endl;
}

 运行结果可知:MyInter 变量 myint 自增一次得到了正确的值。

3.1 重载后置

#include<iostream>
using namespace std;
class MyInter {
	friend ostream& operator<<(ostream& cout, MyInter myint);
public:
	MyInter(){
		m_Num= 0;
	}
	MyInter& operator++() {
		m_Num++; // 先进行++运算
		return *this;
	}

	MyInter operator++(int a) {
		// 先记录 当前值
		MyInter temp = *this;
		// 后递增
		m_Num++;
		// 最后将结果返回
		return temp;
	}
private:
	int m_Num;
};

ostream& operator<<(ostream& cout, MyInter myint) {
	cout << myint.m_Num;
	return cout;
}

int main() {
	MyInter myint;
	cout << myint++ << endl;
	cout << myint;
}

 

4:赋值运算符 

C++ 编译器会至少给一个类添加4个函数

1:默认构造函数(无参,函数体为空)

2:默认析构函数(无参,函数体为空)

3:默认拷贝函数,对属性进行值拷贝

4:赋值运算符 operator= , 对属性进行值拷贝

赋值运算符重载时(重载=),要进行 深拷贝,而不是直接将值进行复制。

#include<iostream>
using namespace std;

class Person {
public:
	int* m_Age;
	Person(int age) {
		// 将年龄数据开辟到堆区
		m_Age = new int(age);
	}
	~Person()
	{
		if (m_Age != NULL)
		{
			delete m_Age;
			m_Age = NULL;
		}
	}

	void operator=(Person& p) {
		if (m_Age != NULL)
		{
			// 先释放掉自己开辟的内存
			delete m_Age;
			m_Age = NULL;
		}
		m_Age = new int(*p.m_Age);
	}
};

int main() {
	Person p1(10);
	Person p2(20);
	p2 = p1;
	cout << "p1的年龄为: " << *p1.m_Age << endl;
	cout << "p2的年龄为: " << *p2.m_Age << endl;
	return 0;
}

运行结果可知:是深拷贝。 

这里有个注意点:如果是出现这种连等情况(右边的那个数赋值给左边),那么就需要 对重载函数 进行修改 ,让其返回当前对象 (return *this)

#include<iostream>
using namespace std;

class Person {
public:
	int* m_Age;
	Person(int age) {
		// 将年龄数据开辟到堆区
		m_Age = new int(age);
	}
	~Person()
	{
		if (m_Age != NULL)
		{
			delete m_Age;
			m_Age = NULL;
		}
	}

	Person& operator=(Person& p) {
		if (m_Age != NULL)
		{
			// 先释放掉自己开辟的内存
			delete m_Age;
			m_Age = NULL;
		}
		m_Age = new int(*p.m_Age);
		return *this;
	}
};

int main() {
	Person p1(10);
	Person p2(20);
	Person p3(30);
	p3 = p2 = p1;
	cout << "p1的年龄为: " << *p1.m_Age << endl;
	cout << "p2的年龄为: " << *p2.m_Age << endl;
	cout << "p3的年龄为: " << *p3.m_Age << endl;
	return 0;
}

 

5:  关系运算符重载 

关系运算符包含 == 和 !=  ,如果现在我们想对比两个自定义的数据类型,则需要重载关系运算符。

案例:假如现在我们有个Person类,包含一个 string类型成员变量name 和 int 类型成员变量age,如果两个 Person对象的 name 和age想对就打印相等,否则打印不相等。

#include<iostream>
#include<string>
using namespace std;
class Person {
public:
	string name;
	int age;
	bool operator==(Person p) {
		if (this->age == p.age && this->name == p.name)
		{
			return true;
		}
		else
		{
			return false;
		}
	}

	bool operator!=(Person p) {
		if (this->age != p.age || this->name != p.name)
		{
			return true;
		}
		else
			return false;
	}

	Person(int age, string name) {
		this->age = age;
		this->name = name;
	}
};

int main() {
	Person p1(10, "tom");
	Person p2(20, "tom");
	if (p1 == p2)
	{
		cout << "p1和p2是相等的" << endl;
	}
	if (p1 != p2)
	{
		cout << "p1和p2是不相等的" << endl;
	}
	return 0;
}

运行结果:也是符号我们预期的 

6:函数调用运算符重载 

1:函数调用运算符() 也可以 重载

2:由于重载后使用的方式非常像函数的调用,因此称为仿函数

3:仿函数没有固定写法,比较 灵活

案例:我们现在创建一个 MyPrint类,通过重载函数调用运算符完成字符串打印输出

#include<iostream>
#include<string>
using namespace std;

class MyPrint {
public:
	void operator()(string str) {
		cout << str << endl;
	}
};

int main() {
	MyPrint myFunc;
	myFunc("hello function operator override");
}

 

 运行结果表明:可以看出类似函数一样完成了字符串的打印输出。

案例:假设现在有一个MyAdd类,通过重载函数调用运算符完成两个整数相加。

#include<iostream>
#include<string>
using namespace std;

class MyAdd {
public:
	int operator()(int a,int b) {
		return a + b;
	}
};

int main() {
	MyAdd myFunc;
	cout << myFunc(10, 20) << endl;
	// 通过匿名对象调用
	cout << MyAdd()(20,30) << endl;
}

 

 匿名函数调用:即先通过 MyAdd()创建一个匿名对象,这个匿名对象在当前指向结束后会被释放,然后为这个匿名对象调用了重载的()运算符函数。

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

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

相关文章

西瓜书-支持向量机

支持向量机 支持向量:距离超平面最近的这几个训练样本点。 支持向量机的核心思想是最大化间隔γ2∣∣w∣∣\gamma \frac{2}{||w||}γ∣∣w∣∣2​。 求极大转换为求极小&#xff0c;转化为凸规划问题。 对偶问题 利用拉格朗日乘子法&#xff0c;对于不等式约束&#xff0c;…

安全网络身份认证系统的设计与实现

本文章源码地址&#xff1a;https://gitee.com/sukels/shirohttps://gitee.com/sukels/shiro 摘 要 随着互联网的飞速发展,Web应用的安全问题日益凸显。为了保护Web应用中用户和企业的敏感信息,认证授权加密已经成为了Web应用中不可缺少的部分。但是随之而来是巨大的代码工作…

欧科云链半年报解读Final

欧科云链半年报解读&#xff1a;净利润同比扭亏为盈&#xff0c;区块链大数据等创新业务贡献新增长曲线 日前&#xff0c;欧科云链控股&#xff08;01499.HK&#xff09;发布2022年度截至9月30日止的六个月中期报告。报告显示&#xff0c;公司在2022年4月1日至2022年9月30日实现…

【idea插件】EasyCode介绍与使用

1.简介 EasyCode是idea的一个插件,主要功能是代码生成,类似的插件还有jpa support EasyCode是idea的一个插件&#xff0c;可以采用图形化的方式对数据的表生成entity,controller,service,dao,mapper……无 需任何编码&#xff0c;简单而强大。可以大幅度的提高开发效率.下面来…

UG/NX二次开发Siemens官方NXOPEN实例解析—2.7 DiameterSymbol

列文章目录 UG/NX二次开发Siemens官方NXOPEN实例解析—2.1 AssemblyViewer UG/NX二次开发Siemens官方NXOPEN实例解析—2.2 Selection UG/NX二次开发Siemens官方NXOPEN实例解析—2.3 Selection_UIStyler UG/NX二次开发Siemens官方NXOPEN实例解析—2.4 File2Points UG/NX二次…

第三章:远程登陆Linux系统-[实操篇]

一&#xff1a;为什么需要远程登陆Linux 1.1示意图 1.2说明 说明: 公司开发时候&#xff0c; 具体的情况是这样的 1) linux 服务器是开发小组共享的. 2) 正式上线的项目是运行在公网的. 3) 因此程序员需要远程登录到 centos 进行项目管理或者开发. 4) 画出简单的网络拓扑…

Vue3——路由的query参数和命名路由以及默认插槽slot的使用

这里主要在message页面组件和detail页面组件介绍 看一个案例&#xff0c;当一个二级路由下面又有许多个不同的跳转页面的时候&#xff0c;比如下图的about/message/detail , 需要分别展示多条信息&#xff0c;这里不能给每一条信息都配置一个组件&#xff0c;那样当信息的数量…

RocketMQ-RocketMQ 系统架构以及消息的概念

文章目录一、RocketMQ的消息模型1、RocketMQ的基础消息模型&#xff0c;一个简单的Pub/Sub模型2、RocketMQ 扩展后的消息模型3、RocketMQ的部署模型二、RocketMQ的系统架构2、Consumer3、Name Server3.1、路由注册3.2、路由剔除3.3、路由发现3.4、Client 对 NameServer选择策略…

Spring5框架总结学习(从入门到进阶)-AOP

文章目录AOP1、基本概念2、底层原理3、底层原理实现4、AOP&#xff08;术语&#xff09;5、准备工作6、基于注解实现AOP 1、基本概念 面向切面编程可用对各个业务逻辑各个部分进行隔离 2、底层原理 AOP底层使用动态代理 有2种情况动态代理 有接口 使用JDK动态代理无接口 使…

ffmpeg编译安装

ffmpeg编译安装前言一、下载ffmpeg二、编译安装2.1 Linux编译ffmpeg2.2 Windows编译ffmpeg总结前言 Fmpeg是一套可以用来记录、转换数字音频、视频&#xff0c;并能将其转化为流的开源计算机程序。采用LGPL或GPL许可证。它提供了录制、转换以及流化音视频的完整解决方案。它包含…

154. SAP UI5 Smart Table 和 Smart Filter Bar 的联合使用方法介绍

本教程第 147 个步骤,我们介绍了 SAP UI5 Smart Table 控件的用法: SAP UI5 应用开发教程之一百四十七 - SAP UI5 SmartTable 控件的使用介绍如下图所示: 本步骤我们在 Smart Table 本身的基础上再进一步,学习如何将 Smart Table 配合 Smart Filter Bar 共同使用。 先看一…

JDK之强软弱虚引用

Java中强软弱虚引用的整体架构&#xff1a; 强引用 当内存不足&#xff0c;JVM开始垃圾回收&#xff0c;对于强引用的对象&#xff0c;就算是出现了OOM也不会对该对象进行回收&#xff0c;死都不收。 强引用是我们最常见的普通对象引用&#xff0c;只要还有强引用指向一个对象…

Fabric.js 元素被遮挡的部分也可以操作

本文简介 点赞 关注 收藏 学会了 题目&#xff1a; 当两个元素有部分重叠时&#xff0c;选中底层元素后&#xff0c;想通过被盖住的部分移动元素&#xff0c;该如何实现&#xff1f; 其实 Fabric.js 已经提供了相应的 API 去完成上面的需求了。但直到今天&#xff0c; Fabr…

动态内存开辟+柔性数组

C/C中程序内存区域划分 &#xff08;数据段就是静态区&#xff09; C/C程序内存分配的几个区域: 1.栈区 (stack): 在执行函数时&#xff0c;函数内局部变量的存储单元都可以在栈上创建&#xff0c;函数执行结束时这些存储单元自动被释放。栈内存分配运算内置于处理器的指令集…

linux系统中裸机实现RTC的基本方法

大家好&#xff0c;今天主要和大家聊一聊&#xff0c;如何显示RTC实时时钟的方法。 目录 第一&#xff1a;RTC实时时钟简介 第二&#xff1a;利用SNVS_LP的SRTC的具体配置方法 第三&#xff1a;初始化RTC对应的代码 第一&#xff1a;RTC实时时钟简介 在高端芯片中&#xff0…

Docker管理面板Crane开源了!

导读数人云容器管理面板 Crane 开源啦&#xff01;Crane 包含着数人云工程师对 Docker 最新技术的热爱和实践。希望借助开源社区的力量&#xff0c;让 Crane 完善自身&#xff0c;更好地成长起来&#xff0c;让更多的国内用户体验并参与到最新的容器技术中。在 Docker 内置 Swa…

Java HashMap

HashMap 是一个散列表&#xff0c;它存储的内容是键值对(key-value)映射。 HashMap 实现了 Map 接口&#xff0c;根据键的 HashCode 值存储数据&#xff0c;具有很快的访问速度&#xff0c;最多允许一条记录的键为 null&#xff0c;不支持线程同步。 HashMap 是无序的&#x…

2023年天津理工大学中环信息学院专业课考试具体安排

天津理工大学中环信息学院2023年高职升本科专业课考试时间地点及防疫须知 一、考试时间地点 考试时间&#xff1a;2022年12月29日上午。 具体考试时间以准考证上显示的为准。 考生可提前半小时入场&#xff0c;考试开始&#xff08;铃声&#xff09;后&#xff0c;禁止入场&…

【python绘制地图——folium实用功能进阶】

Python使用folium制作地图并生成png图片 第一章 folium的方法和类的介绍&#xff08;思维导图&#xff09; 第二章 使用folium制作地图 第三章 folium实用功能进阶 第三章 使用Html2Image生成png图片 第四章 使用reportlab制作pdf报告 文章目录Python使用folium制作地图并生成…

基于C++实现对UNet图像分割的部署

本博文利用UNet介绍图像分割的部署,重点介绍采用warpaffine对图像进行预处理,以及后处理decode部分的代码。 1. Unet网络介绍 UNet它是一个encoder - decoder的结构,那么encoder对应就是这个u型网络左边这半部分,也就是我们特征提取及下采样这部分。decoder解码就是右边部…