鱼C论坛

 找回密码
 立即注册
查看: 5181|回复: 4

关于++--运算在printf()里的

[复制链接]
发表于 2012-11-25 15:34:28 | 显示全部楼层 |阅读模式
10鱼币
这是为什么?
01.png

最佳答案

查看完整内容

首先C语言的传参方式是cdecl 参数从右往左传参 计算也是从右往左 int i; i=5; printf("%d,%d,%d,%d,%d",i=i++,i=i--,i=++i,i=--i,i=i++); 看了下反汇编 发现这个潜规则够没人性.。 先来个总结再讲解 C语言printf对于++ -- 运算符的显示是这样的 首先对于i++ i--这种由于++ --在后面,其意义就是先进行其他运算之后 再自增,面对参数一堆i++,i--的直接无视++ --操作,留到显示完后再 一次性执行一堆++ -- i=i++ 这 ...
小甲鱼最新课程 -> https://ilovefishc.com
发表于 2012-11-25 15:34:29 | 显示全部楼层
本帖最后由 メ㊣逆ご帅☆ 于 2012-11-25 19:14 编辑

首先C语言的传参方式是cdecl
参数从右往左传参
计算也是从右往左
int i;
i=5;
printf("%d,%d,%d,%d,%d",i=i++,i=i--,i=++i,i=--i,i=i++);
看了下反汇编
发现这个潜规则够没人性.。
先来个总结再讲解
C语言printf对于++ -- 运算符的显示是这样的
首先对于i++ i--这种由于++ --在后面,其意义就是先进行其他运算之后
再自增,面对参数一堆i++,i--的直接无视++ --操作,留到显示完后再
一次性执行一堆++ --
i=i++
这句话就是i=i
之后再++
printf("%d,%d,%d,%d,%d",i=i++,i=i--,i=++i,i=--i,i=i++);
前两个++ --最后一个的++留到后面执行
也就是是这样执行的
从右往左计算的
传参的从右到左是不影响显示的从左到右的
1.i赋值自身,最右边的i传进去,也就是显示屏上最右边的5
2.倒数第二个先自减后传进去,i=i-1=4,对应显示4
3.倒数第三个先自增后传进去,i=i+1=5,对应的就是第三个5
4.i赋值自身,第二个直接传进去,显示屏上第二个5
5.i赋值自身,第一个直接传进去,显示屏上第一个5
6.然后i+1
7.i-1
8.i+1
最后i的值应该是6,不过被你覆盖了


对于++i --i是先自减的再显示
++ --放前面后面的意义就是这样的。。
也就是这样的



00401028   mov         dword ptr [ebp-4],5
这句对应i=5
0040102F   mov         eax,dword ptr [ebp-4]
00401032   mov         dword ptr [ebp-4],eax
00401035   mov         ecx,dword ptr [ebp-4]
00401038   mov         dword ptr [ebp-8],ecx
0040103B   mov         edx,dword ptr [ebp-8]
0040103E   push        edx

0040103F   mov         eax,dword ptr [ebp-4]
上面的对应的就是赋值自身后,传进最右边的参数i,这里不进行++ --

00401042   sub         eax,1
从右往左嘛,所以++i --i中第一个被算的是--i
00401045   mov         dword ptr [ebp-4],eax
00401048   mov         ecx,dword ptr [ebp-4]
0040104B   mov         dword ptr [ebp-4],ecx
0040104E   mov         edx,dword ptr [ebp-4]

00401051   push        edx
赋值自身后传进去
00401052   mov         eax,dword ptr [ebp-4]
00401055   add         eax,1

先自增
00401058   mov         dword ptr [ebp-4],eax
0040105B   mov         ecx,dword ptr [ebp-4]
0040105E   mov         dword ptr [ebp-4],ecx
00401061   mov         edx,dword ptr [ebp-4]
00401064   push        edx

赋值自身后传进去
00401065   mov         eax,dword ptr [ebp-4]
00401068   mov         dword ptr [ebp-4],eax
0040106B   mov         ecx,dword ptr [ebp-4]
0040106E   mov         dword ptr [ebp-0Ch],ecx
00401071   mov         edx,dword ptr [ebp-0Ch]
00401074   push        edx

赋值自身后传进去
00401075   mov         eax,dword ptr [ebp-4]
00401078   mov         dword ptr [ebp-4],eax
0040107B   mov         ecx,dword ptr [ebp-4]
0040107E   mov         dword ptr [ebp-10h],ecx
00401081   mov         edx,dword ptr [ebp-10h]
00401084   push        edx

赋值自身后传进去


到这里都传完了

00401085   push        offset string "%d,%d,%d,%d,%d" (0042201c)
0040108A   mov         eax,dword ptr [ebp-4]
0040108D   add         eax,1
00401090   mov         dword ptr [ebp-4],eax
00401093   mov         ecx,dword ptr [ebp-4]
00401096   sub         ecx,1
00401099   mov         dword ptr [ebp-4],ecx
0040109C   mov         edx,dword ptr [ebp-4]
0040109F   add         edx,1
004010A2   mov         dword ptr [ebp-4],edx
上面就是最后的++ --了
004010A5   call        printf (004010f0)
执行显示
004010AA   add         esp,18h

cdecl传参方式,栈由调用者平衡

后面的估计那些估计不用解释我还是编辑解释下好了
i=5;
printf("%d",i++);
先赋值自身(i=i)然后显示i也就是5
之后i=i+1
这时i=6
printf("%d",i--);
先赋值自身(i=i),然后显示i也就是6
之后i=i-1
这时i=5
printf("%d",++i);
先i=i+1
然后i=i
接着显示i,也就是6
这时i=6
printf("%d",--i);
先i=i-1
然后i=i
接着显示i,也就是5
这时i=5
printf("%d",i++);
先赋值自身(i=i),然后显示i也就是5
之后i=i+1
这时i=6


如果你在最后加一个printf应该i会是6,我没试过不知道




小甲鱼老师叫你研究你还真研究了。。。~~o(>_<)o ~~
最后↖(^ω^)↗加油








小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

 楼主| 发表于 2012-11-25 19:50:47 | 显示全部楼层
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

发表于 2012-11-25 19:51:49 | 显示全部楼层
メ㊣逆ご帅☆ 发表于 2012-11-25 18:41
首先C语言的传参方式是cdecl
参数从右往左传参
计算也是从右往左

分析得不错,不过你忽略了一点,虽然C标准规定函数参数是从右短发哦左进栈的,但 进栈之前必须要对各个参数求值。C标准完全没有对函数参数求值顺序作任何规定,所以以上你说的关于函数参数求值顺序仅仅是VC编译器的一种实现而已。这题的 答案也是编译器相关的。

评分

参与人数 1鱼币 +1 收起 理由
メ㊣逆ご帅☆ + 1 感谢。

查看全部评分

小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

发表于 2012-11-27 12:04:59 | 显示全部楼层
メ㊣逆ご帅☆ 发表于 2012-11-25 15:34
首先C语言的传参方式是cdecl
参数从右往左传参
计算也是从右往左

好厉害啊!!!膜拜
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-11-16 12:14

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表