关于自增,自减表达式的作用,我就不多做介绍。这里,罗列出一些不大碰到的问题。
一. 前置,后置的区别,及原因。
对于前置表达式,是通过引用(by-reference)来处理的。也就是,该表达式还回的是变量自身。++i表达式还回的就是&i,自然i中的值是增加过后的值,不然运算结果就会丢失。对于后置表达式,是通过传值(by-value)来实现的。也就是用一个临时变量来保存结果,然后还回这个临时变量。所以i++的结果是i还没有变化,要到一个完整语句(出现‘;或者函数调用)后才变化。
说了这么多废话,无外乎三个结论。
1,前置++,--会将计算的结果保存到变量里面,而后置却将计算结果保存到一个临时变量;
2,既然前置返回的是变量本身,那么他可以做左值,而后置是临时变量,临时变量是const的,就只能做右值。如(++a)++正确,(a++)++不能通过编译。
3,后置会在表达式结束时才将这个临时变量(也就是计算结果)保存到变量里面。这里的表达式结束指一个赋值操作完成,或者一个函数调用(不是函数执行完返回后)后。实际汇编语言是在计算完表达式后,再进行加1/减1操作。但是对于复合运算符+=、-=、*=、/=、%=不适用。比如int x = 10; x -=x--; 结果为-1。因为该式要马上使用x,x必须有个确切值,相当于y = x--; 此时y为10,x为9。x=x- y;就是9-10 所以为 -1。
二. 前置,后置在复合表达式中。
例如: int j, i= 3;
j = (i++)+(i++)+(++i);
计算后,i=6,j却可能等于12,也可能等于10.这就取决于编译器了。因为在标准中,并没有规定复合表达式中,相同运算等级的运算顺序是从左边开始还是从右边开始。例如这个例子,从左边开始,就是3+3+4 = 10;从右边开始就成了4+4+4=12了。所以,除非你是在出考试试卷,否则不要在你的程序中出现同一个变量两次以上自增/自减操作符。这除了破坏你程序可读性,增加维护难度外,没有多少帮助。
三。重载++/--操作符。
根据上面的分析,在做++/--操作符重载时,注意返回值的类型。例如:如果自定义了一个类A(含有一个数据成员x)。
那么对于前置,首先就是对它的数据成员++/--,然后再返回它的引用,这样,才是左值表达式,如下面的代码:
A& A::operator++()
{
++x; //--x
return *this ;
}
对于后置,则如下:显然还回的是一个A的临时实例。而且,初始化值是未增加的x,结束后x加了1.
A A::operator++(int )
{
return A(this->x++);
}