++、--、+=等运算符常被用在循环中的累积运算,但这些看似简单的运算符如果运用不好很容易导致程序出现难易发现的bug。
i++、++i和 i+=
自加和自减运算符一般在循环中用来控制循环次数。例如
for (int i = 0; i < 10; i++)
上面是我们的常用写法,但是还可以有下面两种写法:
for (int i = 0; i < 10; ++i); for (int i = 0; i < 10; i+=1);
这三种方式分别会执行几次循环体呢?
答案是:都是10次;到这里看起来三者好像没有区别对吧。
看一下下面这一段代码:
int i = 0 ; int a = i++ ; cout<<a; i = 0 ; int b = ++i; cout<<b; i = 0 ; int c = i += 1 ; cout<<c;
运行结果a是0,而b是1,c是1;
原因分析
i++是先赋值,然后再自增;++i是先自增,后赋值。
用代码表示就是:
若 a = i++; 则等价于 a=i;i=i+1;
而 b = ++i; 则等价于 i=i+1;b=i;
int c = i += 1 ;牵涉到运算符优先级问题;
运算符优先级
=号和+=同为第14级,但是都是从右向左运算,所以先计算i += 1,然后在赋值给c;
进阶
看下面这段代码:
int i = 0 ; ++i = 10 ; cout<<i ;
输出结果是10,这是为什么呢?i++ = 10 ;又是否可以呢?只有++i可以作为左值i++是不能作为左值得,原因得从运算符的实现原理上来看:
++i实现:
int& int::operator++() //返回一个引用,函数返回值可以作为一个左值 { //函数无参,在自身空间内增加1 *this += 1; // 增加 return *this; // 取回值 }
i++实现:
const int int::operator++(int) //函数返回值是一个非左值型的。 { //函数带参,另外的空间开辟 int oldValue = *this; // 取回值 ++(*this); // 增加 return oldValue; // 返回被取回的值 }