㈠ 字元║在c++編譯之後輸出U是怎麼回事
是的,對於編譯器來說,名字是不一樣而且唯一的,編譯器會對函數名「修飾」。
C++ 編譯器的函數名修飾規則
函數名字修飾(Decorated Name)方式
函數的名字修飾(Decorated Name)就是編譯器在編譯期間創建的一個字元串,用來指明函數的定義或原型。LINK程序或其他工具有時需要指定函數的名字修飾來定位函數的正確位置。多數情況下程序員並不需要知道函數的名字修飾,LINK程序或其他工具會自動區分他們。當然,在某些情況下需要指定函數的名字修飾,例如在C++程序中,為了讓LINK程序或其他工具能夠匹配到正確的函數名字,就必須為重載函數和一些特殊的函數(如構造函數和析構函數)指定名字裝飾。另一種需要指定函數的名字修飾的情況是在匯編程序中調用C或C++的函數。如果函數名字,調用約定,返回值類型或函數參數有任何改變,原來的名字修飾就不再有效,必須指定新的名字修飾。C和C++程序的函數在內部使用不同的名字修飾方式,下面將分別介紹這兩種方式。
1. C編譯器的函數名修飾規則
對於__stdcall調用約定,編譯器和鏈接器會在輸出函數名前加上一個下劃線前綴,函數名後面加上一個「@」符號和其參數的位元組數,例如_functionname@number。__cdecl調用約定僅在輸出函數名前加上一個下劃線前綴,例如_functionname。__fastcall調用約定在輸出函數名前加上一個「@」符號,後面也是一個「@」符號和其參數的位元組數,例如@functionname@number
2. C++編譯器的函數名修飾規則
C++的函數名修飾規則有些復雜,但是信息更充分,通過分析修飾名不僅能夠知道函數的調用方式,返回值類型,參數個數甚至參數類型。不管__cdecl,__fastcall還是__stdcall調用方式,函數修飾都是以一個「?」開始,後面緊跟函數的名字,再後面是參數表的開始標識和按照參數類型代號拼出的參數表。對於__stdcall方式,參數表的開始標識是「@@YG」,對於__cdecl方式則是「@@YA」,對於__fastcall方式則是「@@YI」。參數表的拼寫代號如下所示:
X--void
D--char
E--unsigned char
F--short
H--int
I--unsigned int
J--long
K--unsigned long(DWORD)
M--float
N--double
_N--bool
U--struct
....
指針的方式有些特別,用PA表示指針,用PB表示const類型的指針。後面的代號表明指針類型,如果相同類型的指針連續出現,以「0」代替,一個「0」代表一次重復。U表示結構類型,通常後跟結構體的類型名,用「@@」表示結構類型名的結束。函數的返回值不作特殊處理,它的描述方式和函數參數一樣,緊跟著參數表的開始標志,也就是說,函數參數表的第一項實際上是表示函數的返回值類型。參數表後以「@Z」標識整個名字的結束,如果該函數無參數,則以「Z」標識結束。下面舉兩個例子,假如有以下函數聲明:
int Function1 (char *var1,unsigned long);
其函數修飾名為「?Function1@@YGHPADK@Z」,而對於函數聲明:
void Function2();
其函數修飾名則為「?Function2@@YGXXZ」 。
對於C++的類成員函數(其調用方式是thiscall),函數的名字修飾與非成員的C++函數稍有不同,首先就是在函數名字和參數表之間插入以「@」字元引導的類名;其次是參數表的開始標識不同,公有(public)成員函數的標識是「@@QAE」,保護(protected)成員函數的標識是「@@IAE」,私有(private)成員函數的標識是「@@AAE」,如果函數聲明使用了const關鍵字,則相應的標識應分別為「@@QBE」,「@@IBE」和「@@ABE」。如果參數類型是類實例的引用,則使用「AAV1」,對於const類型的引用,則使用「ABV1」。下面就以類CTest為例說明C++成員函數的名字修飾規則:
1
2
3
4
5
6
7
8
9
10
11
12
class CTest
{
......
private:
void Function(int);
protected:
void CopyInfo(const CTest &src);
public:
long DrawText(HDC hdc, long pos, const TCHAR* text, RGBQUAD color, BYTE bUnder, bool bSet);
long InsightClass(DWORD dwClass) const;
......
};
對於成員函數Function,其函數修飾名為「?Function@CTest@@AAEXH@Z」,字元串「@@AAE」表示這是一個私有函數。成員函數CopyInfo只有一個參數,是對類CTest的const引用參數,其函數修飾名為「?CopyInfo@CTest@@IAEXABV1@@Z」。DrawText是一個比較復雜的函數聲明,不僅有字元串參數,還有結構體參數和HDC句柄參數,需要指出的是HDC實際上是一個HDC__結構類型的指針,這個參數的表示就是「PAUHDC__@@」,其完整的函數修飾名為「?DrawText@CTest@@QAEJPAUHDC__@@JPBDUtagRGBQUAD@@E_N@Z」。InsightClass是一個共有的const函數,它的成員函數標識是「@@QBE」,完整的修飾名就是「?InsightClass@CTest@@QBEJK@Z」。