区别:
i++ 返回的值是新建的一个临时的tmp值,保存的是i增加前的值
,
实际的i值已经被改变了,任何类似*(i++)的操作都会报错,因为i++创建的这个临时值,很快(快到你刚刚获取到i的值)就会被内存删掉,取地址将会导致非法内存引用。
++i 返回 的值是i本身取地址后增加的值
,所以会比i++返回的值大1。
int i = 0;
int *p1 = &(++i); //正确
int *p2 = &(i++); //错误
++i = 1; //正确
i++ = 5; //错误
原理:
他们底层实现不同,
我们都知道,如果在函数内部定义临时变量,return 后,任何对临时变量的访问都是非法的
,
这里对(i++)的地址的访问就是这样,i++实际是用临时的值保存原值,然后返回了这个临时值。实际上的 i的地址上,已经是自增1后的结果了。
而++i则就在原本的地址上自增了1,所以可以取地址。
/ 前缀形式:
int& int::operator++() //这里返回的是一个引用形式,就是说函数返回值也可以作为一个左值使用
{//函数本身无参,意味着是在自身空间内增加1的
*this += 1; // 增加
return *this; // 取回值
}
//后缀形式:
const int int::operator++(int) //函数返回值是一个非左值型的,与前缀形式的差别所在。
{//函数带参,说明有另外的空间开辟
int oldValue = *this; // 取回值
++(*this); // 增加
return oldValue; // 返回被取回的值
}
应用:
理论上说,应用时,为了保证效率,尽量用++i,因为不会创建一个临时变量。
由于现代的编译器还挺高级的,从汇编的角度,i++和++i看不出差别,为了代码的可读性,可以直接用i++;
习题:
学以致用1:
下面的for循环A,B,sum最后的值是多少?
//A
int sum=0;
for(int i=100;i<=100;++i){
sum+=i;
)
//B
int sum=0;
for(int i=100;i<=100;i++){
sum+=i;
)
学以致用2:
i,j的值分别为多少?
int sum=0;
int i= ++sum;
int j= sum++;
学以致用3:
i,j的值分别为多少?
int sum=0;
int i= --sum;
int j= sum--;
答案:
1:A和B中sum的值都为0,
实际上for循环执行的内容是-100加到100(包含-100)
for的逻辑是,
测试条件,若为真,执行循环里的内容,然后执行第三个位置
的自增条件;
若为假,跳出循环,
所以这里,i++还是++i,对for循环来说根本无所谓,因为执行完第三个位置的条件后,去读的都是i位置的值。不存在一个返回值来给左边赋值的情况(例如sum=i++;
)
2:
i=1;
j=1;
和上面同理,
这里,j的值由于是从sum++返回的,sum还没自增,就先返回了一个临时值的原本的值,是还没被更改前的,所以j获取到的值是1。
可能有些小朋友不相信,你自己跑下代码:
#include<iostream>
int main(){
int sum=0;
int i= ++sum;
int j= sum++;
std::cout<<i<<" "<<j<<std::endl;
}
3:
i=-1;
j=-1;