admin 管理员组

文章数量: 888297

printf(“%d,%d\n“,i

// sample.cpp#include<iostream>
using namespace std;int main(){int i=1;printf("Case 1:	%d,%d\n",i--,i++);i=1;printf("Case 2:	%d,%d\n", i++, ++i);i=1;printf("Case 3:	%d,%d,%d\n", i++, ++i, ++i );return 0;
}

运行结果:

 

原因:

参考链接:

 

---

首先看几种情况

1、

int i=1;
printf("%d,%d\n",i--,i++);
运行结果为:2,1

这与编译器有关,通过汇编可以很清楚的看到

第一步:把i的值存入缓存器[ebp-0E8h]=1;

第二步:i值加1,i=i+1=2;

第三步:把i的值存入缓存器[ebp-0ECh]=2;

第四步:i值减1,i=i-1=1;

第五步:把缓存器[ebp-0E8h]=1,入栈

第六步:把缓存器[ebp-0ECh]=2,入栈

 

故,打印输出为 2,1

 

总结:

由++或者--运算的顺序是从右向左,故先计算i++,i++在计算过程中会产生缓存区,返回的值就是缓存区的值,既是在加1之前需要先备份,这里的缓存区地址就是[ebp-0E8h]=1,

之后,i=i+1=2;

同理,第二个表达式的缓存区[ebp-0ECh]=2,

之后,i=i-1=1;

然后把第一,第二个表达式的返回值分别入栈【1,2】故输出为 2 1;

2、

int i=1;
printf("%d,%d\n",i++,++i);

 

第一步:i值加1,i=i+1=2;

第二步:把i的值存入缓存区[ebp-0E8h]=2;

第三步:i值加1,i=i+1=3;

第四步:把i=3,入栈

第五步:把缓存区[ebp-0E8h]=2,入栈

故,输出为2,3

总结:

由++或者--运算的顺序是从右向左,故先计算++i,++i的返回值为i本身,本应该是2,但是后面的运算却影响i的值,另外printf输出流的缓存栈是在所有表达式计算完后再入栈的,只需要知道首先入栈的是i的地址里面存的值,

下面该表达式i++,i++在计算过程中会产生缓存区,返回的值就是缓存区的值,既是在加1之前需要先备份,这里的备份地址就是[ebp-0E8h]=2,之后i=i+1=3,返回的值为缓存区[ebp-0E8h]=2,第二次入栈的就是第二个表达式的返回值[ebp-0E8h]=2,而第一次入栈的是第一个表达式的返回值i的值(此时已经改为3),故输出栈里面是【3,2】,打印输出 2,3.

 

下面的几个例子是具有两个变量的,道理是类似的,要注意,具有后置++/--的,返回的都是缓存区的值,前置的都是返回变量本身的值,另外本身的值可能受另一个表达式的影响,所以,在没有计算完之前,是不知道i的值的。

1、

int x=2,y=3;
printf("%d,%d\n",(x++)+y,++y);//等价x+++y

 

第一个入栈的是y值本身,而其他表达式对其无影响,故入栈y==4

第二个入栈的是表达式的缓存区的值,为x+y=2+4=6,故入栈 6

输出值为6 4

下面的例子可以自己分析试试,看看反汇编就立刻明白含义了,就不一一说明了

2、

int x=2,y=3;
printf("%d,%d\n",x+y++,++y);


3、

int x=2,y=3;
printf("%d,%d\n",x+y,++y);

 

4、

int x=2,y=3;
printf("%d,%d\n",x+y,y++);
 

 

本文标签: printf(“d dn“ I