C++学习笔记----8、掌握类与对象(六)---- 操作符重载(1)

server/2024/10/9 2:17:01/

        经常在对象上执行如相加,比较,文件传输等操作。例如,spreadsheet只有在可以在上面执行自述运算才有用,比如对整行的单元格求和。所有这些都可以通过重载操作符来完成。

        许多人发现操作符重载的语法复杂而令人迷惑。至少一开始是这样。真实情况是想让事情更简单。在本节你会发现,那并不意味着写类时更简单,只是使用类时更简单。关键点是让新类与内建像int与double这样的类型一样简单:使用+来使对象相加要比记住不管是add()或是sum()这样的成员函数名要容易多了。

        注意:作为服务给客户提供类的操作符重载。

        在这一点上,你可能会想到底哪些操作符可以重载呢?答案是几乎所有--甚至是你没有听说过的。本章会浮光掠影地简单涉及一下:赋值操作符在本章的前面 已经解释过了,而本节要介绍的是基础的算术操作符,缩写的算术操作符,以及比较操作符。重载流插入与释放操作符也比较有用。还有,有一些麻烦但是又比较有趣的东东,一开始你可能并不想用操作符重载来做。在标准库中大量使用操作符重载。以后我们会解释怎么以及什么时候重载其他的操作符。以及标准库方面的内容。

1、例子:SpreadsheetCell的另外内容的实现

        在真正的面向对象的编程风格中,SpreadsheetCell对象应该能够把自己与其它SpreadsheetCell对象相加。将一个单元格与另一个单元格相加生成结果为第三个单元格。这并不会改变原来的任一单元格。SpreadsheetCell相加的意思就是单元格值的相加。

1.1、第一次尝试:加成员函数

        可以为SpreadsheetCell类像这样声明与定义add()成员函数:

export class SpreadsheetCell
{
public:SpreadsheetCell add(const SpreadsheetCell& cell) const;// Omitted for brevity
};

        该成员函数将两个单元格相加,返回一个新的第三个单元格 ,其值是前两个的和。声明为const并且 用一个const SpreadsheetCell的引用是因为add()不会改变任意一个源单元格。下面是其实现:

SpreadsheetCell SpreadsheetCell::add(const SpreadsheetCell& cell) const
{return SpreadsheetCell { getValue() + cell.getValue() };
}

        可以像这样来使用add()成员函数:

SpreadsheetCell myCell { 4 }, anotherCell { 5 };
SpreadsheetCell aThirdCell { myCell.add(anotherCell) };
auto aFourthCell { aThirdCell.add(anotherCell) };

        这也可以达到目的,但是有一点儿繁琐,我们可以做得更好。

1.2、第二次尝试:重载操作符+作为成员函数

        用加号类似于两个int或者两个double的方式来将两个单元格相加会比较方便--有点儿像下面这样:

SpreadsheetCell myCell { 4 }, anotherCell { 5 };
SpreadsheetCell aThirdCell { myCell + anotherCell };
auto aFourthCell { aThirdCell + anotherCell };

        c++允许你写自己的加号版本,叫做加操作符,来正确地作用于类。为了做到这一点,写一个名字叫做operator+的成员函数,看起来像这样:

export class SpreadsheetCell
{
public:SpreadsheetCell operator+(const SpreadsheetCell& cell) const;// Omitted for brevity
};

        注意:在operator与加号之间可以加入空格。例如,可以将operator+写成operator +。我们还是使用没有空格的风格。

        重载的operator+成员函数的定义与add()成员函数的实现是一样的:

SpreadsheetCell SpreadsheetCell::operator+(const SpreadsheetCell& cell) const
{return SpreadsheetCell { getValue() + cell.getValue() };
}

        现在你就可以使用前面展示的加号操作符来将两个单元格 相加了。

        我们还是比较习惯于这种语法。不要过于担心奇怪的成员函数名operator+----它就是一个类似于foo或add的名字而已。为了理解语法的其余部分,它有助于理解真实发生的情况。当c++编译器解析程序时,遇到比如说+,-,=,或<<这样的操作符,它会用operator+,operator-,operator=,或operator<<这样的名字来找函数或成员函数,特别是,带有适合的参数的。例如,当编译器看到下面的代码行,它会尝试在SpreadsheetCell类中去找成员函数名字为operator+,接受另一个SpreadsheetCell作为参数(或者,在本章后面会讨论到的,接受两个SpreadsheetCell参数的全局函数,名字叫做operator+):

SpreadsheetCell aThirdCell { myCell + anotherCell };

        如果SpreadsheetCell类包含这样一个operator+成员函数,上面的行就会被翻译成这样:

SpreadsheetCell aThirdCell { myCell.operator+(anotherCell) };

        注意,operator+没有要求其参数的对象是与它所写的类是同一种类型。你可以写一个SpreadsheetCell的operator+,拿一个Spreadsheet去加到SpreadsheetCell。这对于程序员是讲不通的,但是编译器是允许的。下一节会给出一个SpreadsheetCell的operator+接受double值的例子。

        也要注意,可以让operator+返回你想的任何类型。然而,应该遵循最少惊奇的原则;也就是说,operator+的返回类型通常应该是用户想要的。

1.2.1、隐式转换

        令人吃惊的是,一旦写完前面所示的operator+,不但可以将两个单元格相加,也可以将一个单元格加到一个string_view,一个double,或一个int!下面是一些例子:

SpreadsheetCell myCell { 4 }, aThirdCell;
string str { "hello" };
aThirdCell = myCell + string_view{ str };
aThirdCell = myCell + 5.6;
aThirdCell = myCell + 4;

        这些代码能跑的原因是编译器会比尝试找到合适的operator+的所指的正确的类型更多的工作。编译器也会尝试找到合适的类型转换。SpreadsheetCell类有转换构造函数将double或string_view转换成一个SpreadsheetCell。在前面的例子中,当编译器看到一个SpreadsheetCell尝试将自身加到一个double上,它发现SpreadsheetCell构造函数用了double并且 构建了一个临时的SpreadsheetCell对象传递给operator+。同样的,当编译器看到代码行尝试将一个SpreadsheetCell加到一个string_view上时,它会调用string_view SpreadsheetCell构造函数生成一个临时的SpreadsheetCell传递给operator+。

        要记住,虽然使用隐式转换构造函数可以效率不高,因为要生成临时对象。在这个例子中,为了避免隐式构建与double相加,可以写如下的第二个operator+:

SpreadsheetCell SpreadsheetCell::operator+(double rhs) const
{return SpreadsheetCell { getValue() + rhs };
}

http://www.ppmy.cn/server/129042.html

相关文章

智能医疗:Spring Boot医院管理系统开发

2相关技术 2.1 MYSQL数据库 MySQL是一个真正的多用户、多线程SQL数据库服务器。 是基于SQL的客户/服务器模式的关系数据库管理系统&#xff0c;它的有点有有功能强大、使用简单、管理方便、安全可靠性高、运行速度快、多线程、跨平台性、完全网络化、稳定性等&#xff0c;非常适…

c语言实例 -- 循环链表

要求&#xff1a; 创建一个循环链表 下面是一个简单的循环链表&#xff08;循环单链表&#xff09;的C语言实现。循环链表是指链表的最后一个节点的指针指向第一个节点&#xff0c;形成一个环形结构。这种结构的优点是在一些特定的场景可以简化操作逻辑。 循环链表的基本操作…

蓝桥杯—STM32G431RBT6(IIC通信--EEPROM(AT24C02)存储器进行通信)

一、什么是IIC&#xff1f;24C02存储器有什么用&#xff1f; IIC &#xff08;IIC 是半双工通信总线。半双工意味着数据在某一时刻只能沿一个方向传输&#xff0c;即发送数据的时候不能接收数据&#xff0c;接收数据的时候不能发送数据&#xff09;即集成电路总线&#xff08;…

MQ高级:RabbitMQ小细节

在之前的学习中&#xff0c;我们只介绍了消息的发送&#xff0c;但是没有考虑到异常的情况&#xff0c;今天我们就介绍一些异常情况&#xff0c;和细节的部分。 目录 生产者可靠性 生产者重连 生产者确认 MQ可靠性 持久化 Lazy Queue 消费者可靠性 消费者确认机制 失…

Spring Boot:打造下一代医院管理系统

3系统分析 3.1可行性分析 通过对本医院管理系统实行的目的初步调查和分析&#xff0c;提出可行性方案并对其一一进行论证。我们在这里主要从技术可行性、经济可行性、操作可行性等方面进行分析。 3.1.1技术可行性 本医院管理系统采用JAVA作为开发语言&#xff0c;Spring Boot框…

【项目安全设计】软件系统安全设计规范和标准(doc原件)

1.1安全建设原则 1.2 安全管理体系 1.3 安全管理规范 1.4 数据安全保障措施 1.4.1 数据库安全保障 1.4.2 操作系统安全保障 1.4.3 病毒防治 1.5安全保障措施 1.5.1实名认证保障 1.5.2 接口安全保障 1.5.3 加密传输保障 1.5.4终端安全保障 资料获取&#xff1a;私信或者进主页。…

Python知识点:如何使用Google Cloud IoT与Python进行边缘计算

开篇&#xff0c;先说一个好消息&#xff0c;截止到2025年1月1日前&#xff0c;翻到文末找到我&#xff0c;赠送定制版的开题报告和任务书&#xff0c;先到先得&#xff01;过期不候&#xff01; 如何使用Google Cloud IoT与Python进行边缘计算 边缘计算作为一种新兴的计算模式…

软件工程的详细学习要点和学习方向

软件工程的详细学习要点和学习方向主要包括以下几个方面&#xff1a; 学习要点 1. 编程语言基础&#xff1a; - 熟练掌握至少一种编程语言&#xff0c;如Java、Python、C等。 - 学习编程语言的语法、特性、常用库函数&#xff0c;并具备编写、调试和优化代码的能力。 …