(1) _stdcall調(diào)用
_stdcall 是Standard Call的縮寫,是C++的標(biāo)準(zhǔn)調(diào)用方式,也是Pascal程序的缺省調(diào)用方式,參數(shù)采用從右到左的壓棧方式,如果是調(diào)用類成員的話,最后一個(gè)入棧的是this指針。這些堆棧中的參數(shù)由被調(diào)函數(shù)自身在返回前清空,
使用的指令是 retn X,X表示參數(shù)占用的字節(jié)數(shù),CPU在ret之后自動彈出X個(gè)字節(jié)的堆??臻g。稱為自動清棧。
WIN32 Api都采用_stdcall調(diào)用方式,這樣的宏定義說明了問題:
#define WINAPI _stdcall
按C編譯方式,_stdcall調(diào)用約定在輸出函數(shù)名前面加下劃線,后面加“@”符號和參數(shù)的字節(jié)數(shù),形如_functionname@number。
(2) _cdecl調(diào)用
_cdecl是C Declaration[聲明]的縮寫,表示C/C++語言默認(rèn)的函數(shù)調(diào)用方法,也是C/C++的缺省調(diào)用方式,所有參數(shù)從右到左依次入棧,這些參數(shù)由調(diào)用者清除,稱為手動清棧。_cedcl約定的函數(shù)只能被C/C++調(diào)用,每一個(gè)調(diào)用它的
函數(shù)都包含清空堆棧的代碼,所以產(chǎn)生的可執(zhí)行文件大小會比調(diào)用_stdcall函數(shù)的大。被調(diào)用函數(shù)無需要求調(diào)用者傳遞多少參數(shù),調(diào)用者傳遞過多或者過少的參數(shù),甚至完全不同的參數(shù)都不會產(chǎn)生編譯階段的錯(cuò)誤。
由于_cdecl調(diào)用方式的參數(shù)內(nèi)存棧由調(diào)用者維護(hù),所以變長參數(shù)的函數(shù)能(也只能)使用這種調(diào)用約定。
由于Visual C++默認(rèn)采用_cdecl 調(diào)用方式,所以VC中中調(diào)用DLL時(shí),用戶應(yīng)使用_stdcall調(diào)用約定。
按C編譯方式,_cdecl調(diào)用約定僅在輸出函數(shù)名前面加下劃線,形如_functionname。
(3) _fastcall調(diào)用
_fastcall 是編譯器指定的快速調(diào)用方式。由于大多數(shù)的函數(shù)參數(shù)個(gè)數(shù)很少,使用堆棧傳遞比較費(fèi)時(shí)。因此_fastcall通常規(guī)定將前兩個(gè)(或若干個(gè))參數(shù)由寄存器傳遞,其余參數(shù)還是通過堆棧傳遞。不同編譯器編譯的程序規(guī)定的寄
存器不同。返回方式和_stdcall相當(dāng)。
_fastcall調(diào)用較快,它通過CPU內(nèi)部寄存器傳遞參數(shù)。
(4)PASCAL調(diào)用
PASCAL 是Pascal語言的函數(shù)調(diào)用方式,也可以在C/C++中使用,參數(shù)壓棧順序與_cdecl和_stdcall相反。
歸納如下:
調(diào)用約定 入棧參數(shù)清理 參數(shù)入棧順序
----------- -------------- ----------------
cdecl 調(diào)用者處理 右->左
stdcall 函數(shù)自己處理 右->左
fastcall 函數(shù)自己處理 依賴于編譯器
pascal 函數(shù)自己處理 左->右