当前位置: 首页 > article >正文

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

        经常在对象上执行如相加,比较,文件传输等操作。例如,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.kler.cn/news/337202.html

相关文章:

  • Conda答疑
  • 人工智能新闻和发展 (24001)- By 10/4/2024
  • 一天认识一个硬件之交换机
  • 边缘人工智能(Edge Intelligence)
  • 基于Vue3+pinia+vue-router+axios+element-plus等开发的新闻发布管理系统
  • 【Java】springboot 项目中出现中文乱码
  • 基于pytorch的手写数字识别-训练+使用
  • 【ubuntu】ubuntu20.04安装conda
  • 【树莓派5B】IO串口通信使用
  • 软件测试面试题大全
  • JVM 内存布局
  • 用Python Turtle绘制天安门技术讲解
  • Python酷库之旅-第三方库Pandas(136)
  • 使用 Spring Boot 客户端对 Apache Pulsar 进行自定义身份验证
  • 【EXCEL数据处理】000017 案例 保姆级教程,附多个操作案例。EXCEL Match和Index函数。
  • 柯桥生活口语学习之在化妆品店可以用到的韩语句子
  • Selenium 和 Playwright 两者的区别与选择
  • 如何在各大地图平台上标注店铺定位?
  • IDEA的lombok插件不生效了?!!
  • 后门攻击-Python案例