C语言初探 之 printf压栈顺序(printf("%d %d %d %d %d %d\n",a++, ++a, a++, ++a, a++, ++a ))

写在篇头:

在不同的编译器下结果不同,本例测试的环境为 devcpp5.3.0.1 。   

——————————————————————————————————————————————————————————————————————

  先看一段代码,猜猜会是什么输出:

[cpp]  view plain copy print ?
  1. x=1;   printf("%d %d\n",x,x++);  
  2. x=1;   printf("%d %d\n",x++,x);  
  3. x=1;   printf("%d %d %d\n",x,x++,x);  
  4. x=1;   printf("%d %d %d %d\n",x,++x,x++,x);  
     如是~:
[html]  view plain copy print ?
  1. 2 1  
  2. 1 2  
  3. 2 1 2  
  4. 3 3 1 3  

      看到这里,你会发现,有时看起来是从左往右算,有时候是从右往左算,有时候还是从乱序算。。是不是有些郁闷呢。

      其实,在处理printf时,压栈顺序为从右往左,也就是说从右往左的计算(“计算”不等于“输出”)。

      在计算时,遇到x++会记录此时的x的值作为最后的输出结果。遇到x和++x的时候则不会将此时的计算结果作为最终的输出,只会修改x的值,在最终输出的时候都输出x的值(所以++x和x的结果总是一样的)。

      为什么会是这个样子呢?参见某高手解释吧:

      对于a++的结果,是有ebp寻址函数栈空间来记录中间结果的,在最后给printf压栈的时候,再从栈中把中间结果取出来;而对于++a的结果,则直接压寄存器变量,寄存器经过了所有的自增操作。 (来源:http://www.zzzj.com/html/20090609/71613.html)


[小试身手]

[cpp]  view plain copy print ?
  1. a=1;   
  2. printf("%d %d %d %d %d %d\n",a++, ++a, a++, ++a, a++, ++a );  

答案:[ 6 7 4 7 2 7 ]

(选中答案框看结果,你懂的~)

*********************************************************************************************************************************

调用printf()函数,因为此时的printf()函数的参数需要计算(是表达式),计算的方向是从左到右,同时需要保存这些结果。保存的方法是将它们压入堆栈。

    1. a=1;   
    2. printf("%d %d %d %d %d %d\n",a++, ++a, a++, ++a, a++, ++a );  

注意这里的a++和++a计算有些特殊

自增运算常理值: 6     6    4    4    2   2

       a++, ++a, a++, ++a, a++, ++a

a++:                     6                   4                 2         

最后的结果             6         7        4       7        2      7

这几个连续的自增运算使得a最后的值为7。

a++的值在运算过程中就能确定;

而所有++a或者a(如果有)的值是所有对a的值产生影响的运算之后最终的值。

*********************************************************************************************************************************

你可能感兴趣的:(C,Programming,Language)