詳解C++ 動態(tài)庫導出函數(shù)名亂碼及解決
剛接觸C++,在嘗試從 dll 中導出函數(shù)時,發(fā)現(xiàn)導出的函數(shù)名都“亂碼”了。
導出過程如下:
新建一個Win32項目:
新建的解決方案里有幾個導出的示例:
// 下列 ifdef 塊是創(chuàng)建使從 DLL 導出更簡單的
// 宏的標準方法。此 DLL 中的所有文件都是用命令行上定義的 DLLEXPORT_EXPORTS
// 符號編譯的。在使用此 DLL 的
// 任何其他項目上不應定義此符號。這樣,源文件中包含此文件的任何其他項目都會將
// DLLEXPORT_API 函數(shù)視為是從 DLL 導入的,而此 DLL 則將用此宏定義的
// 符號視為是被導出的。
#ifdef DLLEXPORT_EXPORTS
#define DLLEXPORT_API __declspec(dllexport)
#else
#define DLLEXPORT_API __declspec(dllimport)
#endif
// 此類是從 dllExport.dll 導出的
class DLLEXPORT_API CdllExport {
public:
CdllExport(void);
// TODO: 在此添加您的方法。
};
extern DLLEXPORT_API int ndllExport;
DLLEXPORT_API int fndllExport(void);
于是我什么都不做,直接生成,并且在C#里導入看看能否調用,嗯……錯誤來了:
找不到入口點?難道是沒導出么?我們用“Dependency Walker”來看看:
Oh, shit, WTF is this? 導出是導出了,不過怎么都亂碼了?
右鍵選擇“Undecorate C++ Functions”之后才出現(xiàn)了真面目:
不過我們的目的是要在C#中使用,而不是用眼睛在 Dependency 里面看??!嗯,既然入口點的名字都變了,要不我們在 C# 中手動指定入口點試試?
不錯,成功了,我們終于可以使用 C++ dll里導出的函數(shù)了。
不過,這些亂碼到底是什么東西?百度一下很輕松地找到了答案:
DLL(動態(tài)庫)導出函數(shù)名亂碼含義
C++編譯時函數(shù)名修飾約定規(guī)則:
__stdcall調用約定:
1、以"?"標識函數(shù)名的開始,后跟函數(shù)名;
2、函數(shù)名后面以"@@YG"標識參數(shù)表的開始,后跟參數(shù)表;
3、參數(shù)表以代號表示:
X--void
D--char
E--unsigned char
F--short
H--int
I--unsigned int
J--long
K--unsigned long
M--float
N--double
_N--bool
....
PA--表示指針,后面的代號表明指針類型,如果相同類型的指針連續(xù)出現(xiàn),以"0"代替,一個"0"代表一次重復;
4、參數(shù)表的第一項為該函數(shù)的返回值類型,其后依次為參數(shù)的數(shù)據(jù)類型,指針標識在其所指數(shù)據(jù)類型前;
5、參數(shù)表后以"@Z"標識整個名字的結束,如果該函數(shù)無參數(shù),則以"Z"標識結束。
其格式為"?functionname@@YG*****@Z"或"?functionname@@YG*XZ",例如
int Test1(char *var1, unsigned long)-----?Test1@@YGHPADK@Zvoid Test2()-----"?Test2@@YGXXZ"
__cdecl調用約定:
規(guī)則同上面的_stdcall調用約定,只是參數(shù)表的開始標識由上面的"@@YG"變?yōu)?@@YA"。
__fastcall調用約定:
規(guī)則同上面的_stdcall調用約定,只是參數(shù)表的開始標識由上面的"@@YG"變?yōu)?@@YI"。如果要用DEF文件輸出一個"C++"類,則把要輸出的數(shù)據(jù)和成員的修飾名都寫入.def模塊定義文件
所以... 通過def文件來導出C++類是很麻煩的,并且這個修飾名是不可避免的
雖然有約定的含義,但這也真夠麻煩的!我不禁想,我們之前導入 User32.dll,Shell32.dll 等等這些動態(tài)庫的函數(shù)的時候,那些EntryPoint沒見這么麻煩啊,怎么回事?還是萬能的百度……“在到處函數(shù)之前加上“extern "C"”就行了!”,我們來試試:
// 下列 ifdef 塊是創(chuàng)建使從 DLL 導出更簡單的
// 宏的標準方法。此 DLL 中的所有文件都是用命令行上定義的 DLLEXPORT_EXPORTS
// 符號編譯的。在使用此 DLL 的
// 任何其他項目上不應定義此符號。這樣,源文件中包含此文件的任何其他項目都會將
// DLLEXPORT_API 函數(shù)視為是從 DLL 導入的,而此 DLL 則將用此宏定義的
// 符號視為是被導出的。
#ifdef DLLEXPORT_EXPORTS
#define DLLEXPORT_API __declspec(dllexport)
#else
#define DLLEXPORT_API __declspec(dllimport)
#endif
// 此類是從 dllExport.dll 導出的
class DLLEXPORT_API CdllExport {
public:
CdllExport(void);
// TODO: 在此添加您的方法。
};
extern "C" DLLEXPORT_API int ndllExport;
extern "C" DLLEXPORT_API int fndllExport(void);
注意和之前對比,最后兩行有變化。編譯生成,運行 C# 項目:
沒有指定 EntryPoint 了,成功!
以上所述是小編給大家介紹的C++ 動態(tài)庫導出函數(shù)名亂碼及解決詳解整合,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對我們網(wǎng)站的支持!
您可能感興趣的文章
- 04-02c語言沒有round函數(shù) round c語言
- 01-10求子數(shù)組最大和的解決方法詳解
- 01-10深入二叉樹兩個結點的最低共同父結點的詳解
- 01-10數(shù)據(jù)結構課程設計- 解析最少換車次數(shù)的問題詳解
- 01-10數(shù)據(jù)結構課程設計-用棧實現(xiàn)表達式求值的方法詳解
- 01-10HDOJ 1443 約瑟夫環(huán)的最新應用分析詳解
- 01-10深入理解C++中常見的關鍵字含義
- 01-10使用C++實現(xiàn)全排列算法的方法詳解
- 01-10如何查看進程實際的內(nèi)存占用情況詳解
- 01-10深入Main函數(shù)中的參數(shù)argc,argv的使用詳解


閱讀排行
本欄相關
- 04-02c語言函數(shù)調用后清空內(nèi)存 c語言調用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對數(shù)函數(shù)的表達式 c語言中對
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調用函數(shù)求fibo C語言調用函數(shù)求
隨機閱讀
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 01-10C#中split用法實例總結
- 01-10使用C語言求解撲克牌的順子及n個骰子
- 08-05織夢dedecms什么時候用欄目交叉功能?
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10SublimeText編譯C開發(fā)環(huán)境設置
- 01-11ajax實現(xiàn)頁面的局部加載
- 01-10delphi制作wav文件的方法
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 04-02jquery與jsp,用jquery


