__stdcall __cdecl的区别

__stdcall __cdecl都是修饰函数的调用方式的关键字。

两种调用方式都是从右向左将参数入栈。

__cdecl是c/c++的默认调用方式,


__stdcall是WindowsAPI函数的调用方式,是这样分析出来的:
1. 在windef.h文件中可以看到如下定义
#define  WINAPI      __stdcall

2. 分析一个WindowsApi函数GetCurrentDirectoryW
在winbase.h中能看到该函数的声明
WINBASEAPI
DWORD
WINAPI
GetCurrentDirectoryW(
    __in DWORD nBufferLength,
    __out_ecount_part_opt(nBufferLength, 
return   +   1 ) LPWSTR lpBuffer
    );
可以看出来WindowsAPI函数是用WINAPI也就是__stdcall来修饰的。
另外还能看到WINBASEAPI这个修饰符,继续追下去。

3. WINBASEAPI的定义
在winbase.h文件中
#if  !defined(_KERNEL32_)
#define  WINBASEAPI DECLSPEC_IMPORT
#else
#define  WINBASEAPI
#endif
可以看出来  假如没有定义_KERNEL32_的话 WINBASEAPI就是DECLSPEC_IMPORT

4. DECLSPEC_IMPORT的定义
在winnt.h文件中
#if  (defined(_M_IX86) || defined(_M_IA64) || defined(_M_AMD64)) && !defined(MIDL_PASS)
#define  DECLSPEC_IMPORT __declspec(dllimport)
#else
#define  DECLSPEC_IMPORT
#endif
DECLSPEC_IMPORT 是 __declspec(dllimport)


从第3个步骤可以知道,假如没有定义过_KERNEL32_的cpp文件包含winbase.h头文件的话,WINBASEAPI最终会被替换为__declspec(dllimport),也就是我们自己的项目中引用winbase.h的时候,winbase.h会声明GetCurrentDirectory为:
__declspec(dllimport) DWORD __stdcall GetCurrentDirectory( params )

OK,这样我们就证明了WindowsAPI是__stdcall的调用形式的,可是为什么windowsapi是这种调用形式而不是__cdecl调用形式呢?

这就牵扯到,函数结束时堆栈是由主调函数进行清空还是由被调函数(在这里WindowsAPI就是被调函数)进行清空,__stdcall是由被调函数在执行结束时对堆栈进行清空的,而__cdecl调用形式是又主调代码对堆栈区进行清空的,由于WindowsAPI的调用是非常频繁的,为了减小我们的编译出来的exe可执行文件的大小,所以WindowsAPI使用__stdcall调用形式而不是__cdecl调用形式。


转载于:https://www.cnblogs.com/coderlee/archive/2008/02/28/1084940.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值