1 知识点
__cdecl
是C Declaration的缩写(declaration,声明),表示C语言默认
的函数调用方法:所有参数从右到左依次入栈,这些参数由调用者清除,称为手动清栈。被调用函数不会要求调用者传递多少参数,调用者传递过多或者过少的参数,甚至完全不同的参数都不会产生编译阶段的错误。
压栈顺序跟C语言标准没关系,跟编译器有关系。不过一般的编译器都默认设置为__cdecl
。
2 VS中设置压栈顺序
首先打开一个工程—>调试—>xxx项目属性—>配置属性—>C/C++—>高级:
3 __cdecl压栈顺序实例
- 明白计算:计算是从右到左计算的
- 栈和寄存器变量:x++,是将计算结果存放到栈空间,最后是要出栈的;而++x和x是将计算结果直接存放到某个寄存器变量中(是同一个),所以计算完最后输出时,++x和x的结果总是相同的。
用个小例子来说明下:
int x=5; printf("%d %d\n",x,x++);
int y=5; printf("%d %d\n",y++,y);
int z=5; printf("%d %d %d\n",z,z++,z);
int w=5; printf("%d %d %d %d\n",w,++w,w++,w);
/*输出结果*/
6 5
5 6
6 5 6
7 7 5 7
【解释说明】
从右至左计算,对于x,x++为5,压入栈中,再是x,为6,放寄存器中,然后输出,则输出6 5;
从右至左计算,对于y,先是计算y,这时值是5,放入寄存器中,然后计算y++,结果还是5,压入栈中,但是寄存器中值变了,为6,最后开始输出。输出的话,是压入栈(y++)中的就输出对应的值,不是压入栈(y)的就都是寄存器中的值。
从右至左计算,对于z,z为5,放寄存器中,z++,还为5,压入栈中,z则为6了,寄存器值改变为6,然后输出,则输出为6 5 6;
同理可得其他的计算结果。
/*下面这个例子如果结果得出正确,说明掌握了*/
int a = 1;
printf("%d %d %d %d %d %d\n",a++, ++a, a++, ++a, a++, ++a );
/*结果*/
11 12 9 12 7 12
4 其他的压栈实例结果
__fastcall
__stdcall
在VS中设置上面两种调用顺序的时候,编译出来提示:
warning C4007: “wmain”: 必须是“__cdecl”
有时间了再去鼓捣其他的编译器吧。