#include<stdio.h> #include"stdlib.h" #include<iostream> using namespace std;
//58 void __cdecl func3(int i, char *szTest) { cout << "szTest的地址:" << &szTest << endl; cout << "sz本身的值:" << (void*)szTest << endl; cout << "i的地址:" << &i << endl;
int k,k2; cout << "k的地址:" << &k << endl; cout << "k2的地址:" << &k2 << endl; cout << "-------------------------------------------------------" << endl; } void __stdcall func4(int i, char *szTest) { cout << "szTest的地址:" << &szTest << endl; cout << "sz本身的值:" << (void*)szTest << endl; cout << "i的地址:" << &i << endl;
int k,k2; cout << "k的地址:" << &k << endl; cout << "k2的地址:" << &k2 << endl; }
int main()
{
int a[2]; cout << &a[0] << endl; cout << &a[1] << endl << endl;
int i = 0x22; char szTest[4] = {'a','b', 'c', 'd'}; cout << &i << endl; cout << (void*)szTest << endl; cout << *szTest << endl; func3(i, szTest); func4(i, szTest);
}
输出:
0012FE98 //这是a[0]地址 0012FE9C //这是a[1]的地址。
0012FE8C 0012FE80 a szTest的地址:0012FDAC //从这个地址和i的地址比较,可知seTest先入栈的。 sz本身的值:0012FE80 i的地址:0012FDA8 k的地址:0012FD98 k2的地址:0012FD8C ------------------------------------------------------- szTest的地址:0012FDAC //采用__stdcall的函数调用,参数也是从右到左进栈的。 sz本身的值:0012FE80 i的地址:0012FDA8 k的地址:0012FD98 k2的地址:0012FD8C
结论:栈的生长方向是从大到小。
__cdecl:表示函数的参数是从右到左入栈。
注意:
1)_cdecl是C语言采用的默认调用方法,它是由主调用函数进行参数压栈并且恢复堆栈 。它的优点是支持printf这样的可变参数调用 。(既然由主调用函数负责压栈和恢复堆栈,主调函数知道参数的个数 ,所以可以支持printf那样的可变参数调用 。)每一个调用它的函数都包含清空堆栈的代码 ,所 以产生的可执行文件大小会比调用_stdcall函数的大 。函数采用从右到左的压栈方式。VC将函数编译后会在函数名前面加上下划线前缀。是MFC缺省调用约定。 2)_stdcall是Pascal程序的缺省调用方式,通常用于Win32 Api中,函数采用从右到左的压栈方式,自己在退出时清空堆栈。VC将函数编译后会在函数名前面加上下划线前缀,在函数名后加上"@"和参数的字节数。
3) 注意,Windows API默认的函数调用方式是__stdcall。所以有这个定义,#define WINAPI __stdcall。
而MFC,C/C++默认调用方式是__cdecl。