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