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

2016-02-19 20:21 18 1 收藏

今天图老师小编要跟大家分享对C++递增(增量)运算符重载的思考,精心挑选的过程简单易学,喜欢的朋友一起来学习吧!

【 tulaoshi.com - 编程语言 】


    在前面的章节中我们已经接触过递增运算符的重载,那时候我们并没有区分前递增与后递增的差别,在通常情况下我们是分别不出++a与a++的差别的,但的确他们直接是存在明显差别的。
  
  先看如下代码: !-- frame contents -- !-- /frame contents -- #include iostream   
  using namespace std;   
     
  int main()   
  {   
      int a=0; 
      ++(++a);//正确,(++a)返回的是左值 
      (a++)++;//错误,(a++)返回的不是左值 
      system("pause"); 
  }   代码中(a++)++编译出错误,返回“++”需要左值的错误,这正是前递增与后递增的差别导致的,那么又是为什么呢?
  
  原因主要是由C++对递增(增量)运算符的定义引发的。
  
  他们之间的差别主要为以下两点:
  
  1、运算过程中,先将对象进行递增修改,而后返回该对象(其实就是对象的引用)的叫前递增(增量)运算。在运算符重载函数中采用返回对象引用的方式编写。
  
  2、运算过程中,先返回原有对象的值,而后进行对象递增运算的叫后递增(增量)运算。在运算符重载函数中采用值返回的方式编写(这也正是前面(a++)++出错误的原因,(a++)返回的不是引用,不能当作左值继续参加扩号外部的++运算),重载函数的内部实现必须创建一个用于临时存储原有对象值的对象,函数返回的时候就是返回该临时对象。  那么在编写运算符重载函数的时候我们该如何区分前递增运算符重载函数与后递增运算符重载函数呢?
  
  方法就是:在后递增运算符重载函数的参数中多加如一个int标识,标记为后递增运算符重载函数。
  
  具体见如下实例(例一为非成员方式,例二为成员方式)://例一 
   
  //程序作者:管宁         
  //站点: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); 
      couta.aendl; 
      cout"观察后递增情况下临时存储对象的值状态:"(a++).aendl;//这里正是体现后递增操作先返回原有对象值地方 
      couta.aendl; 
      (a++)++; 
      couta.aendl;//由于后递增是值返回状态,所以(a++)++只对a做了一次递增操作,操作后为104而非105。 
      system("pause"); 
  } 更多内容请看C/C++技术专题专题,或 //例二 
   
  //程序作者:管宁         
   !-- frame contents -- !-- /frame contents -- //站点: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); 
  
       couta.aendl; 
      cout"观察后递增情况下临时存储对象的值状态:"(a++).aendl;//这里正是体现后递增操作先返回原有对象值地方 
      couta.aendl; 
      (a++)++; 
      couta.aendl;//由于后递增是值返回状态,所以(a++)++只对a做了一次递增操作,操作后为104而非105。 
      system("pause"); 
  }  通过对前后递增运算的分析,我们可以进一步可以了解到,对于相同情况的单目运算符重载我们都必须做好这些区别工作,保证重载后的运算符符合要求。  更多内容请看C/C++技术专题专题,或

来源:http://www.tulaoshi.com/n/20160219/1623437.html

延伸阅读
C#支持的位逻辑运算符如表2.9所示。运算符号意义运算对象类型运算结果类型对象数实例~位逻辑 非 运算整型,字符型整型1~a&位逻辑 与 运算2a & b|位逻辑 或 运算2a | b^位逻辑 异或 运算2a ^ b 位 左移 运算2a4 位 右移 运算2a2  1、 位逻辑非运算 位逻辑非运算是单目的,只有一个运算对象。位逻辑非运算按位对运算对象的值进行...
移位运算符面向的运算对象也是二进制的“位”。可单独用它们处理整数类型(主类型的一种)。左移位运算符()能将运算符左边的运算对象向左移动运算符右侧指定的位数(在低位补0)。“有符号”右移位运算符()则将运算符左边的运算对象向右移动运算符右侧指定的位数。“有符号”右移位运算符使用了“符号扩展”:若值为正,则在高位插入0;若值...
标签: 电脑入门
所谓逻辑运算符也就是与、或、是、非、真、假等等条件判断符号。这里介绍六条。 1.AND 用途:所有参数的逻辑值为真时返回TRUE(真);只要有一个参数的逻辑值为假,则返回FALSE(假)。 语法:AND(logical1,logical2,…)。 参数:Logical1,logical2,…为待检验的1~30个逻辑表达式,它们的结论或为TRUE(真)或为FALSE(假)。参数必须是逻...
标签: Web开发
今天看一个JS的脚本,发现里面有一句话是这样子写的 var obj = document.getElementById("btn1") || document.getElementById("btn2"); 我觉得这个很奇怪,去查了一下||符号的意思,官方的说法是,逻辑或运算,我理解的应该是返回TRUE或FALSE吧,可以神奇的是JAVSCRIPT居然把可以找到的对象返了回来,我做了个试验,在页面上放一个按...
整数间执行除法运算时,要使用""而不是"/"。"/"运算符要求返回一个单一数值,所以,表面上看似简单的一行代码: C=A/B 实际上包含了3个隐含的转换操作:2个为除法运算做准备,从Integer转换到Single;一个完成最后的赋值操作,从Integer转换到Single。但是如果使用了""操作符,情况就大不相同了!不仅不会有这么多中间步骤,...

经验教程

748

收藏

74
微博分享 QQ分享 QQ空间 手机页面 收藏网站 回到头部