1. 程式人生 > >C++知識回顧之__stdcall、__cdcel和__fastcall三者的區別

C++知識回顧之__stdcall、__cdcel和__fastcall三者的區別

進行 pan span number 適用於 編譯器 api num 處理

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

  1. 調用協議常用場合
    1. __stdcall:Windows API默認的函數調用協議。
    2. __cdecl:C/C++默認的函數調用協議。
    3. __fastcall:適用於對性能要求較高的場合。
  2. 函數參數入棧方式
    1. __stdcall:函數參數由右向左入棧。
    2. __cdecl:函數參數由右向左入棧。
    3. __fastcall:從左開始不大於4字節的參數放入CPU的ECX和EDX寄存器,其余參數從右向左入棧。
    4. 問題一:__fastcall在寄存器中放入不大於4字節的參數,故性能較高,適用於需要高性能的場合。
  3. 棧內數據清除方式
    1. __stdcall:函數調用結束後由被調用函數清除棧內數據。
    2. __cdecl:函數調用結束後由函數調用者清除棧內數據。
    3. __fastcall:函數調用結束後由被調用函數清除棧內數據。
    4. 問題一:不同編譯器設定的棧結構不盡相同,跨開發平臺時由函數調用者清除棧內數據不可行。
    5. 問題二:某些函數的參數是可變的,如printf函數,這樣的函數只能由函數調用者清除棧內數據。
    6. 問題三:由調用者清除棧內數據時,每次調用都包含清除棧內數據的代碼,故可執行文件較大。
  4. C語言編譯器函數名稱修飾規則
    1. __stdcall:編譯後,函數名被修飾為“[email protected]”。
    2. __cdecl:編譯後,函數名被修飾為“_functionname”。
    3. __fastcall:編譯後,函數名給修飾為“@[email protected]”。
    4. 註:“functionname”為函數名,“number”為參數字節數。
    5. 註:函數實現和函數定義時如果使用了不同的函數調用協議,則無法實現函數調用。
  5. C++語言編譯器函數名稱修飾規則
    1. __stdcall:編譯後,函數名被修飾為“?[email protected]@YG******@Z”。
    2. __cdecl:編譯後,函數名被修飾為“?[email protected]@YA******@Z”。
    3. __fastcall:編譯後,函數名被修飾為“?[email protected]@YI******@Z”。
    4. 註:“******”為函數返回值類型和參數類型表。
    5. 註:函數實現和函數定義時如果使用了不同的函數調用協議,則無法實現函數調用。
    6. C語言和C++語言間如果不進行特殊處理,也無法實現函數的互相調用。

C++知識回顧之__stdcall、__cdcel和__fastcall三者的區別