printf压栈问题[保姆级教程]

前言

当printf遇上了i++和++i
当我看到这个代码的时候,突然就对printf和i++产生了疑问。

i=1;    printf("%d %d\n",i,i++)                  // 2 1
i=1;    printf("%d %d\n",i++,i);                 // 1 2
i=1;    printf("%d %d %d\n",i,i++,i);            // 2 1 2
i=1;    printf("%d %d %d %d %d\n",i,++i,i++,i);  // 3 3 1 3
1234

输出的结果是不是有点意外!!!
我查阅了很多资料终于弄明白了。。。

先说下i++和++i的区别

  • i++ 会先复制一个副本,然后在真值上 (真值就是原来的数据) +1,相当于产生了两个变量,一个是副本不会变,一个原来的数据会+1。
  • ++i 则是直接在原数据上+1,不会产生新的中间变量。

然后说下printf的压栈问题

  • 用一个例子讲解一下,

    先自己思考一下最后会输出多少,会能更好的理解

    ,输出结果最后解密,最后的结果可能超出你的预料哟!!!

    int a = 1;

    printf("%d %d %d %d %d %d %d\n", a++,++a,a++,++a,a,a++,++a);
    12

  • printf是先从右向左压栈,压栈的时候瞬间算出结果,然后按这个例子结尾的++a属于高位,开头的a++属于低位,出栈的时候从低位开始出。

    入栈顺序 入栈参数 操作 栈内情况 a的值
    ⑦(低位) a++ 先copy一个a,然后对原数据a+1,将copy的压栈 a(copy3) a a(copy2) a a a(copy1) a 7
    ++a 直接对a+1 a a(copy2) a a a(copy1) a 6
    a++ 先copy一个a,然后对原数据a+1,将copy的压栈 a(copy2) a a a(copy1) a 5
    ++a 直接对a+1 a a a(copy1) a 4
    a 直接就是a a a(copy1) a 3
    a++ 先copy一个a,然后对原数据a+1,将copy的压栈 a(copy1) a 3
    ①(高位) ++a 直接对a+1 a 2

注意: 1,当有副本的时候会输出副本,但是真值已经加了1了。
    2,对真值操作就相当于对原来的a操作,所以每次更改真值,会把栈底的a也会改变。

答案揭晓 !!!
正确答案就是 6 7 4 7 7 2 7.

总结

  • printf是从右向左读取参数压栈,在压栈的时候会算出来值。
  • 改变真值时候是会影响下面的a,其实就是改变原来的数据。
  • 可能老师教i++和++i的时候,是说i++先判断,++i是先加,这让更通俗,但是会把人带坏,记住他们的本质。

给我一个赞,就是对我最大的鼓励。谢谢大家了!!!

THE END