【计算机系统】微妙的 i++和++i——底层实现,原理和应用for循环,i赋值

区别:

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,所以可以取地址。

参考链接:为什么(i++)不能做左值,而(++i)可以

/ 前缀形式:
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;

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值