C++ 运算符重载 (备查)

基础

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

运算符重载也可以发生函数重载

语法:

void operator@();	//@代表了被重载的运算符。函数的参数个数取决于两个因素。1)运算符是一元(一个参数)的还是二元(两个参数)2)运算符被定义为全局函数(对于一元是一个参数,对于二元是两个参数)还是成员函数(对于一元没有参数,对于二元是一个参数-此时该类的对象用作左耳参数)
成员函数重载 + 全局函数重载 原理

在这里插入图片描述

运算符重载(operator overloading)只是一种”语法上的方便”,也就是它只是另一种函数调用的方式。

是否可重载

在这里插入图片描述

几乎C中所有的运算符都可以重载,但运算符重载的使用时相当受限制的。

不能改变运算符优先级,不能改变运算符的参数个数。

加号(+)运算符重载

//利用成员函数实现加号运算符重载
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;
}private: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;
}

重载左移操作符(<<),使得 cout 可以输出自定义对象

为什么不用 成员函数重载 实现

在这里插入图片描述

使用 全局函数 重载 << 运算符(无法链式编程)

在这里插入图片描述

cout 是 ostream 类的对象,全局只有一个,所以使用 & 。

要实现 cout 链式输出需返回 cout 引用

在这里插入图片描述

如果要访问类中私有属性,配合友元实现
//重载 << 运算符案列
class Person{friend ostream& operator<<(ostream& os, Person& person);//友元
public:Person(int id,int age){mID = id;mAge = age;}
private:int mID;int mAge;
};//重载 <<  运算符
ostream& operator<<(ostream& os, Person& person){os << "ID:" << person.mID << " Age:" << person.mAge;return os;
}int main(){Person person(1001, 30);//cout << person; //cout.operator+(person)cout << person << " | " << endl;return EXIT_SUCCESS;
}

自增自减(++/–)运算符重载

class Complex{public:Complex(){mA = 0;mB = 0;}//重载前置++Complex& operator++(){mA++;mB++;return *this;	//返回的的是引用,可以实现链式编程:(++(++b))}//重载后置++Complex operator++(int){	Complex temp;temp.mA = this->mA;temp.mB = this->mB;mA++;mB++;return temp;	//temp是局部对象,不能返回引用,所以不能实现链式编程。}//前置--Complex& operator--(){mA--;mB--;return *this;}//后置--Complex operator--(int){Complex temp;temp.mA = mA;temp.mB = mB;mA--;mB--;return temp;}private:int mA;int mB;
};

如何区分重载的前置++还是后置++,通过占位参数来区分,有 int 的为后置++,- - 同理。

要优先使用前缀形式,由于前缀形式创建了一个临时对象,效率经常会略高一些。

指针运算符( *、-> )重载

利用智能指针,管理 new 出来的 Person 的释放操作。

//被维护的类
class Person{
public:Person(int param){this->mParam = param;}void PrintPerson(){cout << "Param:" << mParam << endl;}
private:int mParam;
};//智能指针
class SmartPointer{
public:SmartPointer(Person* person){this->pPerson = person;}//重载指针的 -> 操作符Person* operator->(){return pPerson;}//重载指针的 * 操作符Person& operator*(){return *pPerson;}~SmartPointer(){if (pPerson != NULL){delete pPerson;	//在析构中释放 被维护对象空间this.pPerson = NULL;}}
public:Person* pPerson;	//维护一个被管理类的指针
};void test01(){SmartPointer pointer(new Person(18));	//通过构造,实例化被被维护的对象//本质:pointer->->PrintPerson(); 编译器简化为:pointer->PrintPerson();pointer->PrintPerson();	(*pointer).PrintPerson();
}

智能指针类 执行在栈上,执行完后自动释放,同时也释放了被维护对象的空间。

自动释放体现出:智能
指针运算符( * ,-> )重载体现出: 指针

赋值(=)运算符重载

编译器 默认给一个类提供四个函数:默认构造,拷贝构造(浅拷贝),析构函数 ,operator= (浅拷贝)。

在这里插入图片描述

//赋值(=)运算符重载-------详细案列
class Person{friend ostream& operator<<(ostream& os,const Person& person){os << "ID:" << person.mID << " Age:" << person.mAge << endl;return os;}
public:Person(int id,int age){this->mID = id;this->mAge = age;}//重载赋值运算符Person& operator=(const Person& person){this->mID = person.mID;this->mAge = person.mAge;return *this;}
private:int mID;int mAge;
};//1. =号混淆的地方
void test01(){Person person1(10, 20);Person person2 = person1; //调用拷贝构造//如果一个对象还没有被创建,则必须初始化,也就是调用构造函数//上述例子由于person2还没有初始化,所以会调用构造函数//由于person2是从已有的person1来创建的,所以只有一个选择//就是调用拷贝构造函数person2 = person1; //调用operator=函数//由于person2已经创建,不需要再调用构造函数,这时候调用的是重载的赋值运算符
}
//2. 赋值重载案例
void test02(){Person person1(20, 20);Person person2(30, 30);cout << "person1:" << person1;cout << "person2:" << person2;person2 = person1;cout << "person2:" << person2;
}
//常见错误,当准备给两个相同对象赋值时,应该首先检查一下这个对象是否对自身赋值了
//对于本例来讲,无论如何执行这些赋值运算都是无害的,但如果对类的实现进行修改,那么将会出现差异;
//3. 类中指针
class Person2{friend ostream& operator<<(ostream& os, const Person2& person){os << "Name:" << person.pName << " ID:" << person.mID << " Age:" << person.mAge << endl;return os;}
public:Person2(char* name,int id, int age){this->pName = new char[strlen(name) + 1];strcpy(this->pName, name);this->mID = id;this->mAge = age;}
#if 1//重载赋值运算符Person2& operator=(const Person2& person){//注意:由于当前对象已经创建完毕,那么就有可能pName指向堆内存//这个时候如果直接赋值,会导致内存没有及时释放if (this->pName != NULL){delete[] this->pName;}this->pName = new char[strlen(person.pName) + 1];strcpy(this->pName,person.pName);this->mID = person.mID;this->mAge = person.mAge;return *this;}
#endif//析构函数~Person2(){if (this->pName != NULL){delete[] this->pName;}}
private:char* pName;int mID;int mAge;
};void test03(){Person2 person1("John",20, 20);Person2 person2("Edward",30, 30);cout << "person1:" << person1;cout << "person2:" << person2;person2 = person1;cout << "person2:" << person2;
}

等于和不等于(==、!=)运算符重载

class Complex{
public:Complex(char* name,int id,int age){this->pName = new char[strlen(name) + 1];strcpy(this->pName, name);this->mID = id;this->mAge = age;}//重载==号操作符bool operator==(const Complex& complex){if (strcmp(this->pName,complex.pName) == 0 && this->mID == complex.mID && this->mAge == complex.mAge){return true;}return false;}//重载!=操作符bool operator!=(const Complex& complex){if (strcmp(this->pName, complex.pName) != 0 || this->mID != complex.mID || this->mAge != complex.mAge){return true;}return false;}~Complex(){if (this->pName != NULL){delete[] this->pName;}}
private:char* pName;int mID;int mAge;
};
void test(){Complex complex1("aaa", 10, 20);Complex complex2("bbb", 10, 20);if (complex1 == complex2){ cout << "相等!" << endl; }if (complex1 != complex2){ cout << "不相等!" << endl; }
}

函数调用符号()重载——(仿函数)

class Complex{
public:int Add(int x,int y){return x + y;}int operator()(int x,int y){return x + y;}
};
void test01(){Complex complex;cout << complex.Add(10,20) << endl;//对象当做函数来调用cout << complex(10, 20) << endl;//使用匿名对象 用完立即释放cout << complex()(10, 20) << endl;
}

不要重载&&、||

不能重载 operator&& 和 operator|| 的原因是,无法在这两种情况下实现内置操作符的完整语义。(即:不能实现短路特性

符号重载建议

在这里插入图片描述

=, [], () 和 -> 操作符只能通过成员函数进行重载

<< 和 >>只能通过全局函数配合友元函数进行重载

不要重载 && 和 || 操作符,因为无法实现短路规则

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

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

相关文章

计算机网络之网络层超详细讲解

个人主页&#xff1a;C忠实粉丝 欢迎 点赞&#x1f44d; 收藏✨ 留言✉ 加关注&#x1f493;本文由 C忠实粉丝 原创 计算机网络之网络层超详细讲解 收录于专栏【计算机网络】 本专栏旨在分享学习计算机网络的一点学习笔记&#xff0c;欢迎大家在评论区交流讨论&#x1f48c; …

嵌入式驱动开发详解6(RTC)

文章目录 前言RTC简介RTC驱动分析RTC驱动框架RTC驱动实现 RTC应用后续 前言 实时时钟是很常用的一个外设&#xff0c;通过实时时钟我们就可以知道年、月、日和时间等信息。 因此在需要记录时间的场合就需要实时时钟&#xff0c;可以使用专用的实时时钟芯片来完成此功能&#x…

什么是MAC地址?什么是IP地址?IP地址与MAC地址是什么关系?

MAC地址是指Media Access Control Address&#xff0c;媒体访问控制地址。MAC地址被烧录在网络设备的ROM之内&#xff0c; IP地址类似于门牌号码&#xff0c;有了门牌号码&#xff0c;邮差才知道把邮件投送到哪里。 有人新建房屋了&#xff0c;就会分配新的门牌号码&#xff08…

go语言的成神之路-标准库篇-os标准库

一、权限 在操作系统&#xff08;OS&#xff09;中&#xff0c;标准库的权限管理是非常重要的&#xff0c;它确保了不同用户和进程能够安全地访问系统资源。以下是一些常见的权限概念和说明&#xff1a; 1.用户权限 用户ID&#xff08;UID&#xff09;&#xff1a;每个用户在…

ASP.NET|日常开发中连接Sqlite数据库详解

ASP.NET&#xff5c;日常开发中连接Sqlite数据库详解 前言一、安装和引用相关库1.1 安装 SQLite 驱动1.2 引用命名空间 二、配置连接字符串2.1 连接字符串的基本格式 三、建立数据库连接3.1 创建连接对象并打开连接 四、执行数据库操作4.1 创建表&#xff08;以简单的用户表为例…

机器学习:监督学习、无监督学习

1. 引言 机器学习是一种人工智能领域的技术&#xff0c;它旨在让计算机通过学习数据和模式&#xff0c;而不是明确地进行编程来完成任务。 机器学习分为监督学习、无监督学习、半监督学习、强化学习 四种。 ​ 2. 监督学习 2.1 什么是监督学习 定义&#xff1a;根据已有的数…

IEEE T-RO 软体机器人手指状态估计实现两栖触觉传感

摘要&#xff1a;南方科技大学戴建生院士、林间院士、万芳老师、宋超阳老师团队近期在IEEE T-RO上发表了关于软体机器人手指在两栖环境中本体感知方法的论文。 近日&#xff0c;南方科技大学戴建生院士、林间院士、万芳老师、宋超阳老师团队在机器人顶刊IEEE T-RO上以《Propri…

MySQL-DML之数据表操作

文章目录 一. 插入表记录1. 向表中插入部分字段2. 向表中插入所有字段,字段的顺序为创建表时的顺序3. 一次添加多条数据信息 二. 更新表记录1. 更新所有记录的指定字段2. 更新符号条件记录的指定字段 三. 删除表记录1. 按条件删除记录2. 清空记录 四. SQL约束1. 主键约束① 添加…

Exp 智能协同管理系统前端首页框架开发

一、 需求分析 本案例的主要目标是开发一个智能学习辅助系统的前端界面&#xff0c;涵盖以下功能模块&#xff1a; 首页&#xff1a;显示系统的总体概览和关键功能介绍。 班级学员管理&#xff1a;实现班级管理和学员管理。 系统信息管理&#xff1a;管理部门和员工信息。 …

5G中的ATG Band

Air to Ground Networks for NR是R18 NR引入的。ATG很多部分和NTN类似中的内容类似。比较明显不同的是&#xff0c;NTN的RF内容有TS 38.101-5单独去讲&#xff0c;而ATG则会和地面网络共用某些band&#xff0c;这部分在38.101-1中有描述。 所以会存在ATG与地面网络之间的相邻信…

MongoDB与阿里云庆祝合作五周年,展望AI赋能新未来

12月3日&#xff0c;在印尼举行的阿里云合作伙伴大会2024上&#xff0c;MongoDB荣膺阿里云“2024技术创新成就奖”&#xff0c;该奖项旨在表彰与阿里云保持长期稳定合作&#xff0c;通过深度技术融合&#xff0c;在产品技术创新、行业区域深耕等领域取得卓越成就的伙伴。自2019…

未来已来:人工智能如何重塑我们的生活与工作

引言 未来的生活和工作场景正从想象走向现实。想象一下&#xff0c;一个清晨&#xff0c;语音助手已经为你安排好一天的任务&#xff0c;自动驾驶汽车准时送你上班&#xff0c;智能冰箱提醒你需要补充的食材。曾经只存在于科幻小说中的场景&#xff0c;如今正在我们的身边实现。…

苹果全家桶接入ChatGPT,近屿智能邀您共绘AI蓝图

北京时间12月12日凌晨&#xff0c;OpenAI开启了备受瞩目的第五天技术直播。宣布了一个令人振奋的消息&#xff1a;苹果的iPhone、iPad、Mac以及智能助手Siri可以原生使用ChatGPT。 这一合作无疑为生成式AI赛道注入了新的活力&#xff0c;作为全球科技行业的巨头&#xff0c;苹果…

企业级日志分析系统ELK之ELK概述

ELK 概述 ELK 介绍 什么是 ELK 早期IT架构中的系统和应用的日志分散在不同的主机和文件&#xff0c;如果应用出现问题&#xff0c;开发和运维人员想排 查原因&#xff0c;就要先找到相应的主机上的日志文件再进行查找和分析&#xff0c;所以非常不方便&#xff0c;而且还涉及…

安全攻击平台介绍

目录 XSS攻击平台 Attack API BeEF XSS-Proxy 漏洞平台 cnvd 阿里云漏洞库 攻防演练平台 XCTF 攻防平台 零日靶场&#xff08;0ops&#xff09; 安恒靶场&#xff08;赛宁安全&#xff09; XSS攻击平台 XSS Payload如此强大&#xff0c;为了使用方便&#xff0c;有安…

计算机毕业设计Python+Vue.js游戏推荐系统 Steam游戏推荐系统 Django Flask 游 戏可视化 游戏数据分析 游戏大数据 爬虫 机

温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 作者简介&#xff1a;Java领…

【电机控制器】FM33LF015芯片——FLASH模拟EEPROM

【电机控制器】FM33LF015芯片——FLASH模拟EEPROM 文章目录 [TOC](文章目录) 前言一、概述二、代码三、实验四、参考资料总结 前言 使用工具&#xff1a; 1.ARM仿真器/J-OBV2仿真器 提示&#xff1a;以下是本篇文章正文内容&#xff0c;下面案例可供参考 一、概述 二、代码 …

算法日记48 day 图论(拓扑排序,dijkstra)

今天继续图论章节&#xff0c;主要是拓扑排序和dijkstra算法。 还是举例说明。 题目&#xff1a;软件构建 117. 软件构建 (kamacoder.com) 题目描述 某个大型软件项目的构建系统拥有 N 个文件&#xff0c;文件编号从 0 到 N - 1&#xff0c;在这些文件中&#xff0c;某些文件…

物联网安全-ARMv8-M Trustzone 实操

前言 本文针对ARMv8m架构M23/M33 MCU安全特性使用进行介绍,以nxp LPC55xx系列和STM32L5xx系列为例,为大家阐述如何使用Trustzone技术提高物联网设备安全性,适合有一定平台安全基础的物联网设备开发人员、安全方案开发人员。 背景 为了提升平台安全性,ARM推出了ARMv8m架构…

若依集成Uflo2工作流引擎

文章目录 1. 创建子模块并添加依赖1.1 新建子模块 ruoyi-uflo1.2 引入 Uflo2 相关依赖 2. 配置相关 config2.1 配置 ServletConfig2.2 配置 UfloConfig2.3 配置 TestEnvironmentProvider 3. 引入Uflo配置文件4. 启动并访问 Uflo2 是由 BSTEK 自主研发的一款基于 Java 的轻量级工…