#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。
转自:http://hi.baidu.com/lieyu063/item/2ea9a2792527163c714423ad