广告
返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >关于C++运算符重载的一些困惑详解
  • 247
分享到

关于C++运算符重载的一些困惑详解

2024-04-02 19:04:59 247人浏览 安东尼
摘要

一.背景 在复习《c++基础与提高》时,自己实现运算符重载(i++)时,几次都报错。其实还是自己对运算符重载这一部分内容理解得不够透彻,于是再次看了下书上的内容,理解算是加深了一些,

一.背景

在复习《c++基础与提高》时,自己实现运算符重载(i++)时,几次都报错。其实还是自己对运算符重载这一部分内容理解得不够透彻,于是再次看了下书上的内容,理解算是加深了一些,于是提笔记录一下。

环境:win10,Qt4.8

二.概述

这部分内容主要关于在重载函数中,函数前要不要加const,何时加const,返回类型要不要加&(引用)修饰,何时加&(引用)的问题,还有临时对象的问题。关于为什么要重载,重载的规则,友元重载、成员重载的区别之类的知识点,这里就不赘述了。

三.内容

以类Complex为例


class Complex
{
public:
    Complex(double x = 0, double y = 0)
        :m_x(x), m_y(y){}

    void dis()
    {
        cout<<"("<<m_x<<", "<<m_y<<")"<<endl;
    }
protected:
    double m_x;
    double m_y;
};

1.以实现单目运算符prefix++和surfix++为例。

先提这个例子,一是因为我在复习这块时遇到了一点问题,二是这个有点特别,涉及到哑元的问题。

prefixe++

1).考虑基本数据类型,以int类型为例,如下的操作都是可以的;


int a = 1;
++a;
++++a;

2).先实现基本的语义,代码如下:


Complex Complex::operator++(void)
{
    m_x++;
    m_y++;
    return *this;
}

3)考虑添加

重载函数返回的是对象自身,并且需要修改对象,我们即可以想到返回的是引用类型。注意,此时引用指向的对象在重载函数调用时就已经存在了。

4)先运行一下,看下是否能编译通过

++c1;

++++c1;

此时重载函数实现的效果,与基本类型效果一致,符合预期,此时就不考虑重载函数前面是否加const修饰了。


#include <iOStream>

using namespace std;

class Complex
{
public:
    Complex(double x = 0, double y =0)
        :m_x(x), m_y(y){}

    void dis()
    {
        cout<<"("<<m_x<<", "<<m_y<<")"<<endl;
    }

    Complex & operator++(void);
protected:
    double m_x;
    double m_y;
};

Complex & Complex::operator++(void)
{
    m_x++;
    m_y++;
    return *this;
}

int main()
{
    double a = 1.0;
    cout<<++a<<endl;
    ++++a;
    cout<<a<<endl;

    Complex c1(1.0, 2.0);

    Complex cc = ++c1;
    cc.dis();
    cc = ++++c1;  // cc = (c1.operator++()).operator++();
    cc.dis();


    return 0;
}

结果如下

surfix++

为了区分prefix++和surfix++两个成员函数,须使用哑元进行区分(引入 哑元,增加了入参的方式,在调用时不需要添加任何的参数),其实类似一个占位符。

1).考虑基本数据类型,以int类型为例,可以进行的操作和不可以进行的操作


int b = 1;
b++;  // 支持
b++++;  // 不支持

2).先实现基本的语义,代码如下


Complex operator++(int)
    {
        Complex temp = *this;
        m_x++;
        m_y++;
        return temp;
    }

3)考虑添加

可以观察到,重载函数返回的是一个临时对象。若是串联调用,这个临时对象它又会调用一次此重载函数

c1.operator++(0).operator++(0);

调用完,然后就消失了。

此时切不可在返回类型中添加&。原因如下:

【不要返回局部对象的引用或指针】

函数完成后,它所占用的存储空间也随之被释放掉。因此,函数终止意味着局部变量的引用将指向不再有效的内存区域。同样地,函数终止,局部对象被释放,指针将指向一个不存在的对象。

4)先运行一下,看下是否能编译通过

我们会发现,第34行无法通过编译,但是第42行可以通过编译。

5)重载的运算符是否会导致表达式可以被赋值,应该以基础类型为准,如int a, b, c; (a=b)=c;是可以的,而(a+b)=c;是不允许的。返回类型通过加const加以限定来实现。

为了使自定义类型与基本数据类型一致,我们在返回类型前面加上const。重载函数中代码修改为如下


const Complex operator++(int);

修改之后,我们可以看到,第34行和42行均无法通过编译,符合预期。

2.双目运算符+

1)考虑基本类型,以下操作都是支持的


int a1 = 1, a2 = 2, a3 = 3;
int m;
m = a1+a2;
m = a1+(a2+a3);
m = (a1+a2)+a3;

2)先重载=,成员函数如下


Complex & Complex::operator=(const Complex &another)
{
    this->m_x = another.m_x;
    this->m_y = another.m_y;
    return *this;
}

3)再重载运算符+,如下:

因为并未修改传入的参数,所以参数前加了const


Complex Complex::operator+(const Complex &another)
{
    return Complex(this->m_x + another.m_x, this->m_y + another.m_y);
}

4)返回类型是否需要加const呢?

我们再对比下表达式的赋值情况,第49行,对于基本类型,临时对象被赋值的情况编译无法通过,但是第58行,自定义类型却编译通过了。此时,为了使其编译不过,可通过在返回值类型前加const加以限定。

将代码


Complex Complex::operator+(const Complex &another);

修改为如下:


const Complex Complex::operator+(const Complex &another);

5)此时,发现第49和58行均无法通过编译,同时第55行和第57行也编译不过了。

这个是为啥呢?

再仔细看刚修改的代码和第57行代码。重载函数返回类型加了const后,返回的就是const对象了。第57行代码,c1 + c2 + c3; c1 + c2返回的是const对象,而重载函数是一个非const函数。此时,即会报错。

在const修饰类一节中,有学习过:如果const构成函数重载,const对象只能调用const函数,非const对象优先调用非const函数。

调整,在重载函数后面添加const,如下:


const Complex Complex::operator+(const Complex &another) const;

四.结尾

学无止境,继续前行,

参考材料

《C++基础与提高》 王桂林

《C++ Primer》第5版SB、JL、BE

到此这篇关于C++运算符重载的一些困惑的文章就介绍到这了,更多相关C++运算符重载内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: 关于C++运算符重载的一些困惑详解

本文链接: https://www.lsjlt.com/news/124361.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 关于C++运算符重载的一些困惑详解
    一.背景 在复习《C++基础与提高》时,自己实现运算符重载(i++)时,几次都报错。其实还是自己对运算符重载这一部分内容理解得不够透彻,于是再次看了下书上的内容,理解算是加深了一些,...
    99+
    2022-11-12
  • 关于C++的重载运算符和重载函数
    目录C++重载运算符和重载函数C++ 中的函数重载C++ 中的运算符重载可重载运算符/不可重载运算符C++重载运算符和重载函数 C++ 允许在同一作用域中的某个函数和运算符指定多个定...
    99+
    2023-05-19
    C++重载运算符 C++重载函数
  • C++中的运算符重载详解
    目录1、引例2、类中自动建立的函数3、重载赋值运算符解析总结1、引例 class Complex { private: double Real,Image; public: ...
    99+
    2022-11-13
  • 详解C++编程中一元运算符的重载
    可重载的一元运算符如下: !(逻辑“非”) &(取址) ~(二进制反码) *(取消指针引用) +(一元加) -(一元求反) ++(递增) --(递减) 转换运算符 后...
    99+
    2022-06-04
    详解 运算符
  • C++重载的奥义之运算符重载详解
    目录0、引言1、运算符重载定义2、作为成员函数进行重载3、作为全局函数进行重载4、运算符重载的一些规则0、引言 重载,顾名思义从字面上理解就是重复装载,打一个不恰当的比方,你可以用一...
    99+
    2023-05-16
    C++ 运算符重载 C++ 运算符 C++ 重载
  • C语言运算符的重载详解
    目录运算符重载的规则 1. 运算符重载为成员函数2. 运算符重载为非成员函数        ...
    99+
    2022-11-13
  • C++运算符重载的详细讲解
    加号运算符重载 对于内置数据类型,编译器知道如何运算 但是对于自己封装的类,编译器无法进行运算 这时可以通过自己定义运算符重载进行运算 operator+ 通过成员函数重载+号 ...
    99+
    2022-11-12
  • C++双目运算符+=的重载详解
    目录1、+=重载2、friend重载+=3、运算符3.1 单目运算符3.2 双目运算符3.3 三目运算符4、重载++和重载- -总结1、+=重载 class Complex { ...
    99+
    2022-11-12
  • C++中的运算符重载问题详解
    C++中的运算符重载问题详解运算符重载是C++中一个强大而有用的特性,通过重载运算符,可以使得某个类的对象可以像基本类型数据一样使用各种运算符,方便地进行各种操作。本文将详细解释运算符重载的概念,并提供具体的代码示例。在C++中,运算符重载...
    99+
    2023-10-22
    运算符重载 (Operator Overloading) C++中的运算符 (Operators in C++) 重载运
  • 详解C++ 运算符重载中返回值的坑
    相信不少朋友在学习运算符重载的时候,都会被参数与返回值应该是左值引用,还是右值引用,还是const常量所困扰。当然我无法一一枚举,这次先讲一下返回值的坑 (没错就是我亲手写的bug)...
    99+
    2022-11-12
  • 详解C++中递增运算符重载的实现
    目录递增运算符递增运算符重载的实现左移运算符重载前置递增运算符重载后置递增运算符重载递增运算符 形如"a++"、"++a"。如果a初始值为1,...
    99+
    2022-11-13
  • 详解C++中赋值,关系,函数调用运算符重载的实现
    目录赋值运算符重载类结构问题的出现具体实现关系运算符重载类结构具体实现函数调用运算符重载类结构具体实现总结赋值运算符重载 在C++中基本数据类型例如整型,可以实现连续赋值:a=b=c...
    99+
    2022-11-13
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作