天天看點

C/C++ 函數調用方式

來源:http://baike.baidu.com/view/1280676.htm

幾種函數調用方式

  __cdecl 是C Declaration的縮寫(declaration,聲明),表示C語言預設的函數調用方法:所有參數從右到左依次入棧,這些參數由調用者清除,稱為手動清棧。被調用函數不會要求調用者傳遞多少參數,調用者傳遞過多或者過少的參數,甚至完全不同的參數都不會産生編譯階段的錯誤。   _stdcall 是StandardCall的縮寫,是C++的标準調用方式:所有參數從右到左依次入棧,如果是調用類成員的話,最後一個入棧的是this指針。這些堆棧中的參數由被調用的函數在傳回後清除,使用的指令是 retnX,X表示參數占用的位元組數,CPU在ret之後自動彈出X個位元組的堆棧空間。稱為自動清棧。函數在編譯的時候就必須确定參數個數,并且調用者必須嚴格的控制參數的生成,不能多,不能少,否則傳回後會出錯。   PASCAL 是 Pascal語言的函數調用方式,也可以在C/C++中使用,參數壓棧順序與前兩者相反。傳回時的清棧方式與_stdcall相同。   _fastcall是 編譯器指定的快速調用方式。由于大多數的函數參數個數很少,使用堆棧傳遞比較費時。是以_fastcall通正常定将前兩個(或若幹個)參數由寄存器傳遞,其餘參數還是通過堆棧傳遞。不同編譯器編譯的程式規定的寄存器不同。傳回方式和_stdcall相當。   _thiscall 是為了解決類成員調用中this指針傳遞而規定的。_thiscall要求把this指針放在特定寄存器中,該寄存器由編譯器決定。VC使用ecx,Borland的C++編譯器使用eax。傳回方式和_stdcall相當。   _fastcall 和 _thiscall涉及的寄存器由編譯器決定,是以不能用作跨編譯器的接口。是以Windows上的COM對象接口都定義為_stdcall調用方式。   C中不加說明預設函數為_cdecl方式(C中也隻能用這種方式),C++也一樣,但是預設的調用方式可以在IDE環境中設定。   帶有可變參數的函數必須且隻能使用_cdecl方式,例如下面的函數:   int printf(char * fmtStr, ...);   int scanf(char * fmtStr, ...);   */幾種調用約定的差別

幾種調用約定的差別

  __cdecl __fastcall與 __stdcall,三者都是調用約定(Calling convention),它決定以下内容:1)函數參數的壓棧順序,2)由調用者還是被調用者把參數彈出棧,3)以及産生函數修飾名的方法。   1、__stdcall調用約定:函數的參數自右向左通過棧傳遞,被調用的函數在傳回前清理傳送參數的記憶體棧,   2、_cdecl是C和C++程式的預設調用方式。每一個調用它的函數都包含清空堆棧的代碼,是以産生的 可執行檔案大小會比調用_stdcall函數的大。函數采用從右到左的壓棧方式。注意:對于可變參數的成員函數,始終使用__cdecl的轉換方式。   3、__fastcall調用約定:它是通過寄存器來傳送參數的(實際上,它用ECX和EDX傳送前兩個雙字(DWORD)或更小的參數,剩下的參數仍舊自右向左壓棧傳送,被調用的函數在傳回前清理傳送參數的記憶體棧)。   4、thiscall僅僅應用于"C++"成員函數。this指針存放于CX寄存器,參數從右到左壓。thiscall不是關鍵詞,是以不能被程式員指定。   5、nakedcall采用1-4的調用約定時,如果必要的話,進入函數時編譯器會産生代碼來儲存ESI,EDI,EBX,EBP寄存器,退出函數時則産生代碼恢複這些寄存器的内容。naked call不産生這樣的代碼。naked call不是類型修飾符,故必須和_declspec共同使用。

名字修飾約定

  1、修飾名(Decoration name):"C"或者"C++"函數在内部(編譯和連結)通過修飾名識别   2、C編譯時函數名修飾約定規則:   __stdcall調用約定在輸出函數名前加上一個下劃線字首,後面加上一個"@"符号和其參數的位元組數,格式為_f[email protected],例如 :function(int a, int b),其修飾名為:[email protected]   __cdecl調用約定僅在輸出函數名前加上一個下劃線字首,格式為_functionname。   __fastcall調用約定在輸出函數名前加上一個"@"符号,後面也是一個"@"符号和其參數的位元組數,格式為@[email protected]。

設定方法

  1 可以直接在代碼中寫 __cdecl 等調用約定   2 在MS-VC++6.0中,調用約定也可以通過工程設定:Setting...\C/C++ \Code Generation項進行選擇,預設狀态為__cdecl。名字修飾約定。