C++的栈的生长方向和函数调用方式
2011-06-01 09:29
288 查看
#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。
#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。
相关文章推荐
- 分析C++方式构造函数调用虚函数的问题
- C/C++:函数的编译方式与调用约定以及extern “C”的使用
- C++对象模型的那些事儿之六:成员函数调用方式
- C#调用C(C++)dll示例教程,包含dll找不到函数入口的解决办法,包含C#dll调用方式不对出错的解决办法
- C++中函数调用时的三种参数传递方式
- C/C++:函数的编译方式与调用约定以及extern “C”的使用
- C++中函数调用时的三种参数传递方式
- C++ 函数调用方式(_stdcall, _pascal, _cdecl...)总结
- C++中函数调用的方式
- 使用类成员函数指针调用C++成员函数的一种方式
- PASCAL FAR ----c++ 函数的调用方式
- C++中的成员函数调用原理及this指针的传递方式
- C++程序中不同的函数调用方式
- C++基础之函数调用方式
- C/C++不同函数调用方式(在汇编下调试)总结
- C++中函数调用时的三种参数传递方式
- C++中函数调用的方式
- C/C++ 函数调用方式
- C++对象模型的那些事儿之六:成员函数调用方式
- C/C++:函数的编译方式与调用约定以及extern “C”的使用