http://hi.baidu.com/xxhkblog/blog/item/4c3e0430ac74d99da9018e3f.html
_cdecl 是C Declaration的縮寫,表示C語言默認的函數(shù)調(diào)用方法:所有參數(shù)從右到左依次入棧,這些參數(shù)由調(diào)用者清除,稱為手動清棧。被調(diào)用函數(shù)無需要求調(diào)用者傳遞多少參數(shù),調(diào)用者傳遞過多或者過少的參數(shù),甚至完全不同的參數(shù)都不會產(chǎn)生編譯階段的錯誤。
_stdcall 是Standard Call的縮寫,是C++的標準調(diào)用方式:所有參數(shù)從右到左依次入棧,如果是調(diào)用類成員的話,最后一個入棧的是this指針。這些堆棧中的參數(shù)由被調(diào)用的 函數(shù)在返回后清除,使用的指令是 retn X,X表示參數(shù)占用的字節(jié)數(shù),CPU在ret之后自動彈出X個字節(jié)的堆棧空間。稱為自動清棧。函數(shù)在編譯的時候就必須確定參數(shù)個數(shù),并且調(diào)用者必須嚴格的 控制參數(shù)的生成,不能多,不能少,否則返回后會出錯。
PASCAL 是Pascal語言的函數(shù)調(diào)用方式,也可以在C/C++中使用,參數(shù)壓棧順序與前兩者相反。返回時的清棧方式忘記了。。。
_fastcall 是編譯器指定的快速調(diào)用方式。由于大多數(shù)的函數(shù)參數(shù)個數(shù)很少,使用堆棧傳遞比較費時。因此_fastcall通常規(guī)定將前兩個(或若干個)參數(shù)由寄存器傳 遞,其余參數(shù)還是通過堆棧傳遞。不同編譯器編譯的程序規(guī)定的寄存器不同。返回方式和_stdcall相當。
_thiscall 是為了解決類成員調(diào)用中this指針傳遞而規(guī)定的。_thiscall要求把this指針放在特定寄存器中,該寄存器由編譯器決定。VC使用ecx,Borland的C++編譯器使用eax。返回方式和_stdcall相當。
_fastcall 和 _thiscall涉及的寄存器由編譯器決定,因此不能用作跨編譯器的接口。所以Windows上的COM對象接口都定義為_stdcall調(diào)用方式。
C中不加說明默認函數(shù)為_cdecl方式(C中也只能用這種方式),C++也一樣,但是默認的調(diào)用方式可以在IDE環(huán)境中設置。
帶有可變參數(shù)的函數(shù)必須且只能使用_cdecl方式,例如下面的函數(shù):
int printf(char * fmtStr, ...);
int scanf(char * fmtStr, ...);
http://hi.baidu.com/lieyu063/blog/item/aa55fb1bfbaa0dfbae51331c.html
這兩個關鍵字看起來似乎很少和我們打交道,但是看了下面的定義(來自windef.h
),你一定會覺得驚訝:
#define CALLBACK __stdcall
#define WINAPI __stdcall
#define WINAPIV __cdecl
#define APIENTRY WINAPI
#define APIPRIVATE __stdcall
#define PASCAL __stdcall
#define cdecl _cdecl
#ifndef CDECL
#define CDECL _cdecl
#endif
幾乎我們寫的每一個WINDOWS API函數(shù)都是__stdcall類型的,為什么??
首先,我們談一下兩者之間的區(qū)別:
WINDOWS的函數(shù)調(diào)用時需要用到棧(STACK,一種先入后出的存儲結構)。當函數(shù)
調(diào)用完成后,棧需要清除,這里就是問題的關鍵,如何清除??
如果我們的函數(shù)使用了_cdecl,那么棧的清除工作是由調(diào)用者,用COM的術語來講
就是客戶來完成的。這樣帶來了一個棘手的問題,不同的編譯器產(chǎn)生棧的方式不盡相同
,那么調(diào)用者能否正常的完成清除工作呢?答案是不能。
如果使用__stdcall,上面的問題就解決了,函數(shù)自己解決清除工作。所以,在跨
(開發(fā))平臺的調(diào)用中,我們都使用__stdcall(雖然有時是以WINAPI的樣子出現(xiàn))。
那么為什么還需要_cdecl呢?當我們遇到這樣的函數(shù)如fprintf()它的參數(shù)是可變
的,不定長的,被調(diào)用者事先無法知道參數(shù)的長度,事后的清除工作也無法正常的進行
,因此,這種情況我們只能使用_cdecl。
到這里我們有一個結論,如果你的程序中沒有涉及可變參數(shù),最好使用__stdcal
l關鍵字