幾種函式呼叫方式

firedragonpzy發表於2012-07-27


幾種函式呼叫方式

  __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呼叫約定在輸出函式名前加上一個下劃線字首,後面加上一個"@"符號和其引數的位元組數,格式為_functionname@number,例如 :function(int a, int b),其修飾名為:_function@8
  __cdecl呼叫約定僅在輸出函式名前加上一個下劃線字首,格式為_functionname。
  __fastcall呼叫約定在輸出函式名前加上一個"@"符號,後面也是一個"@"符號和其引數的位元組數,格式為@functionname@number。
編輯本段
設定方法

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


摘自:[url]http://baike.baidu.com/view/1280676.htm[/url]

相關文章