首页
学习
活动
专区
圈层
工具
发布
30 篇文章
1
《挑战30天C++入门极限》C++的iostream标准库介绍(1)
2
《挑战30天C++入门极限》C++的iostream标准库介绍(2)
3
《挑战30天C++入门极限》C++的iostream标准库介绍(3)
4
《挑战30天C++入门极限》C++运算符重载赋值运算符
5
《挑战30天C++入门极限》C++运算符重载函数基础及其值返回状态
6
《挑战30天C++入门极限》对C++递增(增量)运算符重载的思考
7
《挑战30天C++入门极限》类的分解,抽象类与纯虚函数的需要性
8
《挑战30天C++入门极限》C++中利用构造函数与无名对象简化运算符重载函数
9
《挑战30天C++入门极限》C++中类的多态与虚函数的使用
10
《挑战30天C++入门极限》C++运算符重载转换运算符
11
《挑战30天C++入门极限》图例实解:C++中类的继承特性
12
《挑战30天C++入门极限》C++类的继承与多重继承的访问控制
13
《挑战30天C++入门极限》图文例解C++类的多重继承与虚拟继承
14
《挑战30天C++入门极限》理解C++面向对象程序设计中的抽象理论
15
《挑战30天C++入门极限》C++面向对象编程入门:构造函数与析构函数
16
《挑战30天C++入门极限》C++类对象的复制-拷贝构造函数
17
《挑战30天C++入门极限》C++类静态数据成员与类静态成员函数
18
《挑战30天C++入门极限》入门教程:实例详解C++友元
19
《挑战30天C++入门极限》C++面向对象编程入门:类(class)
20
《挑战30天C++入门极限》新手入门:C++中的函数重载
21
《挑战30天C++入门极限》新手入门:C/C++中的结构体
22
《挑战30天C++入门极限》C/C++中字符串常量的不相等性及字符串的Copy
23
《挑战30天C++入门极限》新手入门:C++中堆内存(heap)的概念和操作方法
24
《挑战30天C++入门极限》C/C++中字符指针数组及指向指针的指针的含义
25
《挑战30天C++入门极限》C/C++中结构体(struct)知识点强化
26
《挑战30天C++入门极限》新手入门:关于C++中的内联函数(inline)
27
《挑战30天C++入门极限》新手入门:C/C++中枚举类型(enum)
28
《挑战30天C++入门极限》新手入门:C++中布尔类型
29
《挑战30天C++入门极限》新手入门:C/C++中数组和指针类型的关系
30
《挑战30天C++入门极限》新手入门:C++下的引用类型

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

代码语言:javascript
复制






 

对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");  
}

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

 




 




 











下一篇
举报
领券