对C++递增(增量)运算符重载的思考

  在前面的章节中我们已经接触过递增运算符的重载,那时候我们并没有区分前递增与后递增的差别,在通常情况下我们是分别不出++a与a++的差别的,但的确他们直接是存在明显差别的。

  先看如下代码:

#include <iostream>   

using namespace std;   

   
int main()   

{   
    int a=0; 
    ++(++a);//正确,(++a)返回的是左值 
    (a++)++;//错误,(a++)返回的不是左值 
    system("pause"); 
}

  代码中(a++)++编译出错误,返回“++”需要左值的错误,这正是前递增与后递增的差别导致的,那么又是为什么呢?

  原因主要是由C++对递增(增量)运算符的定义引发的。

他们之间的差别主要为以下两点:

  1、运算过程中,先将对象进行递增修改,而后返回该对象(其实就是对象的引用)的叫前递增(增量)运算。在运算符重载函数中采用返回对象引用的方式编写。

  2、运算过程中,先返回原有对象的值,而后进行对象递增运算的叫后递增(增量)运算。在运算符重载函数中采用值返回的方式编写(这也正是前面(a++)++出错误的原因,(a++)返回的不是引用,不能当作左值继续参加扩号外部的++运算),重载函数的内部实现必须创建一个用于临时存储原有对象值的对象,函数返回的时候就是返回该临时对象。

  那么在编写运算符重载函数的时候我们该如何区分前递增运算符重载函数与后递增运算符重载函数呢?

  方法就是://程序作者:管宁         
//站点:www.cndev-lab.com         
//所有稿件均有版权,如要转载,请务必著名出处和作者      
   

#include <iostream>   
using namespace std;   
   

class Test     
{     
    public:     
        Test(int a=0) 
        { 
            Test::a = a; 
        } 
    friend Test& operator ++ (Test&); 
    friend Test operator ++ (Test&,int); 

    public: 
    int a; 

}; 
Test& operator ++ (Test &temp)//前递增 

    temp.a++; 
    return temp; 

Test operator ++ (Test &temp,int)//后递增,int在这里只起到区分作用,事实上并没有实际作用 


    Test rtemp(temp);//这里会调用拷贝构造函数进行对象的复制工作 

    temp.a++; 
    return rtemp; 


int main() 


    Test a(100); 
    ++(++a); 
    cout<<a.a<<endl; 
    cout<<"观察后递增情况下临时存储对象的值状态:"<<(a++).a<<endl;//这里正是体现后递增操作先返回原有对象值地方 
    cout<<a.a<<endl; 
    (a++)++; 

    cout<<a.a<<endl;//由于后递增是值返回状态,所以(a++)++只对a做了一次递增操作,操作后为104而非105。 

    system("pause"); 
}

//例二 
 
//程序作者:管宁         
//站点:www.cndev-lab.com         
//所有稿件均有版权,如要转载,请务必著名出处和作者      
   

#include <iostream>   
using namespace std;   
   

class Test     
{     
    public:     
        Test(int a=0) 
        { 
            Test::a = a; 
        } 
    Test& operator ++ (); 
    Test operator ++ (int); 
    public: 
    int a; 
}; 

Test& Test::operator ++ ()//前递增 

    this->a++; 

    return *this; 


Test Test::operator ++ (int)//后递增 


    Test rtemp(*this);//这里会调用拷贝构造函数进行对象的复制工作 
 
    this->a++; 
    return rtemp; 


int main() 


    Test a(100); 
    ++(++a); 
    cout<<a.a<<endl; 
    cout<<"观察后递增情况下临时存储对象的值状态:"<<(a++).a<<endl;//这里正是体现后递增操作先返回原有对象值地方 
    cout<<a.a<<endl; 
    (a++)++; 

    cout<<a.a<<endl;//由于后递增是值返回状态,所以(a++)++只对a做了一次递增操作,操作后为104而非105。 

    system("pause"); 
}

  通过对前后递增运算的分析,我们可以进一步可以了解到,对于相同情况的单目运算符重载我们都必须做好这些区别工作,保证重载后的运算符符合要求。

 
 

《挑战30天C++入门极限》对C++递增(增量)运算符重载的思考的更多相关文章

  1. 《挑战30天C++入门极限》入门教程:实例详解C++友元

        入门教程:实例详解C++友元 在说明什么是友元之前,我们先说明一下为什么需要友元与友元的缺点: 通常对于普通函数来说,要访问类的保护成员是不可能的,如果想这么做那么必须把类的成员都生命成为pu ...

  2. 《挑战30天C++入门极限》C++面向对象编程入门:构造函数与析构函数

        C++面向对象编程入门:构造函数与析构函数 请注意,这一节内容是c++的重点,要特别注意! 我们先说一下什么是构造函数. 上一个教程我们简单说了关于类的一些基本内容,对于类对象成员的初始化我们 ...

  3. 《挑战30天C++入门极限》C++类静态数据成员与类静态成员函数

        C++类静态数据成员与类静态成员函数 在没有讲述本章内容之前如果我们想要在一个范围内共享某一个数据,那么我们会设立全局对象,但面向对象的程序是由对象构成的,我们如何才能在类范围内共享数据呢? ...

  4. 《挑战30天C++入门极限》C++类对象的复制-拷贝构造函数

        C++类对象的复制-拷贝构造函数 在学习这一章内容前我们已经学习过了类的构造函数和析构函数的相关知识,对于普通类型的对象来说,他们之间的复制是很简单的,例如: int a = 10; int ...

  5. 《挑战30天C++入门极限》C++面向对象编程入门:类(class)

        C++面向对象编程入门:类(class) 上两篇内容我们着重说了结构体相关知识的操作. 以后的内容我们将逐步完全以c++作为主体了,这也意味着我们的教程正式进入面向对象的编程了. 前面的教程我 ...

  6. 《挑战30天C++入门极限》新手入门:C/C++中的结构体

        新手入门:C/C++中的结构体 什么是结构体? 简单的来说,结构体就是一个可以包含不同数据类型的一个结构,它是一种可以自己定义的数据类型,它的特点和数组主要有两点不同,首先结构体可以在一个结构 ...

  7. 《挑战30天C++入门极限》新手入门:C++中堆内存(heap)的概念和操作方法

        新手入门:C++中堆内存(heap)的概念和操作方法 堆内存是什么呢? 我们知道在c/c++中定义的数组大小必需要事先定义好,他们通常是分配在静态内存空间或者是在栈内存空间内的,但是在实际工作 ...

  8. 《挑战30天C++入门极限》新手入门:C++中的函数重载

        新手入门:C++中的函数重载 函数重载是用来iostream>  using namespace std;  int test(int a,int b);  float test(flo ...

  9. 《挑战30天C++入门极限》新手入门:关于C++中的内联函数(inline)

        新手入门:关于C++中的内联函数(inline) 在c++中,为了解决一些频繁调用的小函数大量消耗栈空间或者是叫栈内存的问题,特别的引入了inline修饰符,表示为内联函数. 可能说到这里,很 ...

随机推荐

  1. CAS 5.x搭建常见问题系列(1).未认证的授权服务

    错误内容 错误信息如下: 未认证授权的服务 CAS的服务记录是空的,没有定义服务.希望通过CAS进行认证的应用程序必须在服务记录中明确定义 错误原因 CAS 5.x 默认情况下不支持HTTP的客户端接 ...

  2. VS2019删除大量空白行或者缩进大量空白行

    原文:VS2019删除大量空白行或者缩进大量空白行 问题描述: 在vs编辑器的代码中有时含有大量无用的空白行,我们想删除这些大量空白行或者缩进空白行. 注: 不需要将代码复制在类似word的文本编辑器 ...

  3. SVN上文件出现左侧黄色箭头右侧绿色箭头的双向箭头

    转自:https://blog.csdn.net/jiuweihu521/article/details/90902152 与资源库对比又没有要提交的东西,网上说删除这个目录,然后更新整个配置库..我 ...

  4. vue页面跳转

    一.在template中的常见写法: <router-link to="/recommend"> <button class="button" ...

  5. [破解版]Unity3d引擎最新稳定版本4.5.5下载(官方最新稳定版本)

    来源:http://www.unitymanual.com/thread-28912-1-1.html unity4.5.5 Mac版下载地址:http://pan.baidu.com/s/1hqzi ...

  6. Linux的网络不通流程

    a:xshell连不上的问题第一步:检查网络适配器,是否禁用vmware的虚拟机网卡第二步:检查vmware net8的地址是否为10.0.0.1第三步:检查系统的vmware服务是否启动第四步:检查 ...

  7. IObit Driver Booster 无法更新驱动的解决办法

    IObit Driver Booster 无法更新驱动的解决办法:依次打开软件中的 菜单-设置-网络-自定义代理设置-主机:填入210.101.131.231 端口:8080 最后点确定完成. 注意! ...

  8. 0001-代码仓库-mvn

    暂缺 基本介绍 web管理 ifsvnadmin

  9. LeetCode - 86、分隔链表

    给定一个链表和一个特定值 x,对链表进行分隔,使得所有小于 x 的节点都在大于或等于 x 的节点之前. 你应当保留两个分区中每个节点的初始相对位置. 示例: 输入: head = 1->4-&g ...

  10. 51nod 2497 数三角形

    小b有一个仅包含非负整数的数组a,她想知道有多少个三元组(i,j,k),满足i<j<k且a[i],a[j],a[k]可能作为某个三角形的三条边的边长. 收起   输入 第一行输入一个正整数 ...