C++栈的生长方向和函数传参压栈方式

#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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值