您的位置:首页 > 其它

WINBASEAPI 与WINAPI(__stdcall __cdecl)的区别

2015-04-28 14:36 260 查看
__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调用形式。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: