_cdecl 是C Declaration的縮寫,表示C語(yǔ)言默認(rèn)的函數(shù)調(diào)用方法:所有參數(shù)從右到左依次入棧,這些參數(shù)由調(diào)用者清除,稱為手動(dòng)清棧。被調(diào)用函數(shù)不需要求調(diào)用者傳遞多少參數(shù),調(diào)用者傳遞過多或者過少的參數(shù),甚至完全不同的參數(shù)都不會(huì)產(chǎn)生編譯階段的錯(cuò)誤。
_stdcall 是Standard Call的縮寫,是C++的標(biāo)準(zhǔn)調(diào)用方式:所有參數(shù)從右到左依次入棧,如果是調(diào)用類成員的話,最后一個(gè)入棧的是this指針。這些堆棧中的參數(shù)由被調(diào)用的函數(shù)在返回后清除,使用的指令是 retn X,X表示參數(shù)占用的字節(jié)數(shù),CPU在ret之后自動(dòng)彈出X個(gè)字節(jié)的堆棧空間。稱為自動(dòng)清棧。函數(shù)在編譯的時(shí)候就必須確定參數(shù)個(gè)數(shù),并且調(diào)用者必須嚴(yán)格的控制參數(shù)的生成,不能多,不能少,否則返回后會(huì)出錯(cuò)。
PASCAL 是Pascal語(yǔ)言的函數(shù)調(diào)用方式,也可以在C/C++中使用,參數(shù)壓棧順序與前兩者相反。返回時(shí)的清棧方式忘記了。。。
_fastcall 是編譯器指定的快速調(diào)用方式。由于大多數(shù)的函數(shù)參數(shù)個(gè)數(shù)很少,使用堆棧傳遞比較費(fèi)時(shí)。因此_fastcall通常規(guī)定將前兩個(gè)(或若干個(gè))參數(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對(duì)象接口都定義為_stdcall調(diào)用方式。
C中不加說(shuō)明默認(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, ...);
*/
調(diào)用約定:
__cdecl __fastcall與
__stdcall,三者都是調(diào)用約定(Calling convention),它決定以下內(nèi)容:1)函數(shù)參數(shù)的壓棧順序,2)由調(diào)用者還是被調(diào)用者把參數(shù)彈出棧,3)以及產(chǎn)生函數(shù)修飾名的方法。
1、__stdcall調(diào)用約定:函數(shù)的參數(shù)自右向左通過棧傳遞,被調(diào)用的函數(shù)在返回前清理傳送參數(shù)的內(nèi)存棧,
2、_cdecl是C和C++程序的缺省調(diào)用方式。每一個(gè)調(diào)用它的函數(shù)都包含清空堆棧的代碼,所以產(chǎn)生的可執(zhí)行文件大小會(huì)比調(diào)用_stdcall函數(shù)的大。函數(shù)采用從右到左的壓棧方式。注意:對(duì)于可變參數(shù)的成員函數(shù),始終使用__cdecl的轉(zhuǎn)換方式。
3、__fastcall調(diào)用約定:它是通過寄存器來(lái)傳送參數(shù)的(實(shí)際上,它用ECX和EDX傳送前兩個(gè)雙字(DWORD)或更小的參數(shù),剩下的參數(shù)仍舊自右向左壓棧傳送,被調(diào)用的函數(shù)在返回前清理傳送參數(shù)的內(nèi)存棧)。
4、thiscall僅僅應(yīng)用于"C++"成員函數(shù)。this指針存放于CX寄存器,參數(shù)從右到左壓。thiscall不是關(guān)鍵詞,因此不能被程序員指定。
5、naked call采用1-4的調(diào)用約定時(shí),如果必要的話,進(jìn)入函數(shù)時(shí)編譯器會(huì)產(chǎn)生代碼來(lái)保存ESI,EDI,EBX,EBP寄存器,退出函數(shù)時(shí)則產(chǎn)生代碼恢復(fù)這些寄存器的內(nèi)容。naked call不產(chǎn)生這樣的代碼。naked call不是類型修飾符,故必須和_declspec共同使用。
調(diào)用約定可以通過工程設(shè)置:Setting...\C/C++ \Code Generation項(xiàng)進(jìn)行選擇,缺省狀態(tài)為__cdecl。
名字修飾約定:
1、修飾名(Decoration name):"C"或者"C++"函數(shù)在內(nèi)部(編譯和鏈接)通過修飾名識(shí)別
2、C編譯時(shí)函數(shù)名修飾約定規(guī)則:
__stdcall調(diào)用約定在輸出函數(shù)名前加上一個(gè)下劃線前綴,后面加上一個(gè)"@"符號(hào)和其參數(shù)的字節(jié)數(shù),格式為_functionname@number,例如 :function(int a, int b),其修飾名為:_function@8
__cdecl調(diào)用約定僅在輸出函數(shù)名前加上一個(gè)下劃線前綴,格式為_functionname。
__fastcall調(diào)用約定在輸出函數(shù)名前加上一個(gè)"@"符號(hào),后面也是一個(gè)"@"符號(hào)和其參數(shù)的字節(jié)數(shù),格式為@functionname@number。