[參考]__stdcall、__cdcel和__fastcall三者的區別


一、概述

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

 

二、調用協議常用場合 

__stdcall:Windows API默認的函數調用協議。

 __cdecl:C/C++默認的函數調用協議。 

__fastcall:適用於對性能要求較高的場合。

 

三、 函數參數入棧方式 

__stdcall:函數參數由右向左入棧。

 __cdecl:函數參數由右向左入棧。

 __fastcall:從左開始不大於4字節的參數放入CPU的ECX和EDX寄存器,其余參數從右向左入棧。

 

四、棧內數據清除方式 

__stdcall:函數調用結束后由被調用函數清除棧內數據。 

__cdecl:函數調用結束后由函數調用者清除棧內數據。 

__fastcall:函數調用結束后由被調用函數清除棧內數據。 

 

五、常見問題 

  1. __fastcall在寄存器中放入不大於4字節的參數,故性能較高,適用於需要高性能的場合。 
  2. 不同編譯器設定的棧結構不盡相同,跨開發平台時由函數調用者清除棧內數據不可行。 
  3. 某些函數的參數是可變的,如printf函數,這樣的函數只能由函數調用者清除棧內數據。
  4. 由調用者清除棧內數據時,每次調用都包含清除棧內數據的代碼,故可執行文件較大。 

 

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

__stdcall:編譯后,函數名被修飾為"_functionname@number"。 

__cdecl:編譯后,函數名被修飾為"_functionname"。 

__fastcall:編譯后, 函數名給修飾為"@functionname@nmuber"。 

注:"functionname"為函數名,"number"為參數字節數。 

注:函數實現和函數定義時如果使用了不同的函數調用協議,則無法實現函數調用。 

 

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

__stdcall:編譯后,函數名被修飾為"?functionname@@YG******@Z"。 

__cdecl:編譯后,函數名被修飾為"?functionname@@YA******@Z"。 

__fastcall:編譯后,函數名被修飾為"?functionname@@YI******@Z"。 

注:"******"為函數返回值類型和參數類型表。

 注:函數實現和函數定義時如果使用了不同的函數調用協議,則無法實現函數調用。 C語言和C++語言間如果不進行特殊處理,也無法實現函數的互相調用。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM