天天看點

C++ 之_stdcall、_cdcel和_fastcall三者的差別

__stdcall、__cdecl和__fastcall是三種函數調用協定,函數調用協定會影響函數參數的入棧方式、棧内資料的清除方式、編譯器函數名的修飾規則等。

調用協定常用場合

  1. __stdcall:Windows API預設的函數調用協定。
  2. __cdecl:C/C++預設的函數調用協定。
  3. __fastcall:适用于對性能要求較高的場合。

函數參數入棧方式

  1. __stdcall:函數參數由右向左入棧。
  2. __cdecl:函數參數由右向左入棧。
  3. __fastcall:從左開始不大于4位元組的參數放入CPU的ECX和EDX寄存器,其餘參數從右向左入棧。
  • 問題一:__fastcall在寄存器中放入不大于4位元組的參數,故性能較高,适用于需要高性能的場合。

棧内資料清除方式

  1. __stdcall:函數調用結束後由被調用函數清除棧内資料。
  2. __cdecl:函數調用結束後由函數調用者清除棧内資料。
  3. __fastcall:函數調用結束後由被調用函數清除棧内資料。
  • 問題一:不同編譯器設定的棧結構不盡相同,跨開發平台時由函數調用者清除棧内資料不可行。
  • 問題二:某些函數的參數是可變的,如printf函數,這樣的函數隻能由函數調用者清除棧内資料。
  • 問題三:由調用者清除棧内資料時,每次調用都包含清除棧内資料的代碼,故可執行檔案較大。

C語言編譯器函數名稱修飾規則

  1. __stdcall:編譯後,函數名被修飾為“[email protected]”。
  2. __cdecl:編譯後,函數名被修飾為“_functionname”。
  3. __fastcall:編譯後,函數名給修飾為“@[email protected]”。
  • 注:“functionname”為函數名,“number”為參數位元組數。
  • 注:函數實作和函數定義時如果使用了不同的函數調用協定,則無法實作函數調用。

C++語言編譯器函數名稱修飾規則

  1. __stdcall:編譯後,函數名被修飾為“?functionname@@YG******@Z”。
  2. __cdecl:編譯後,函數名被修飾為“?functionname@@YA******@Z”。
  3. __fastcall:編譯後,函數名被修飾為“?functionname@@YI******@Z”。
  • 注:“******”為函數傳回值類型和參數類型表。
  • 注:函數實作和函數定義時如果使用了不同的函數調用協定,則無法實作函數調用。
  • C語言和C++語言間如果不進行特殊處理,也無法實作函數的互相調用。

繼續閱讀