[DLL - Beginers]使用DEF文件修復函數名——對《使用LoadLibrary調用從Dll中輸出的class》的一點補充 收藏
使用DEF文件修復函數名
——對《使用LoadLibrary調用從Dll中輸出的class》的一點補充
作者 李成竹
在《使用……》一文中,作者在“代碼”的第三點提到了“使用一個DEF文件修復了函數名”,但是并沒有講解什么是DEF文件,也沒有說明應該如何修復,可能會使某些初學者(包括我自己)感到疑惑。我也上網搜索了一下,講解DEF文件作用以及詳細使用方法的文章不多且比較零散,本文在此用一個簡單例子簡單闡述一下DEF文件一般的使用方法,以方便需要者查閱。
?
DEF文件的全稱是Module-Definition File,即模塊定義文件,是用來定義EXE和DLL文件的一種文件格式,以文本形式保存(可用記事本創建/編輯)。由于鏈接器為大多數模塊定義聲明提供了對應的命令行選項,所以一般的Win32程序并不需要.DEF文件。但是在編寫DLL時,尤其是在編寫C++的DLL時,(由于名稱修飾)DEF文件還是有它的用武之地的。
※注:關于“名稱修飾”在很多地方都有介紹,文中不作講解。
?
DEF文件的主要內容是由一系列的聲明(statement)組成,包括NAME、LIBRARY、DISCRIPTION、STACKSIZE、SECTIONS、EXPORTS、VERSION。
?
¨???????? NAME:指定輸出文件的文件名,設置image基址
¨???????? LIBRARY(DLL):指定DLL的內部名稱和加載時的基址
¨???????? DISCRIPTION:文件描述
¨???????? STACKSIZE:設置棧的大小
¨???????? SECTIONS:設置image文件的一個或多個段屬性
¨???????? EXPORTS:定義輸出列表
¨???????? VERSION:指定文件版本
?
其中最常用的是LIBRARY、EXPORTS和DISCRIPTION。
?
示例
1.????? 現在有一個已經編寫好的類要用DLL輸出,并通過函數名對DLL進行動態調用。其頭文件和源文件如下:
Header File:
#ifdef LIBDLL_EXPORTS
#define LIBDLL_API __declspec(dllexport)
#else
#define LIBDLL_API __declspec(dllimport)
#endif
#include <iostream.h>
// This class is exported from the LibDll.dll
class LIBDLL_API CTest
{
? int data;
public:
? CTest();
? void print();
};
Source File:
#include "stdafx.h"
#include "LibDll.h"
BOOL APIENTRY DllMain( HANDLE hModule,
?????????????????????? DWORD? ul_reason_for_call,
?????????????????????? LPVOID lpReserved
???????????????????????????????? )
{
??? switch (ul_reason_for_call)
? {
???????? case DLL_PROCESS_ATTACH:
???????? case DLL_THREAD_ATTACH:
???????? case DLL_THREAD_DETACH:
???????? case DLL_PROCESS_DETACH:
???????????????? break;
??? }
??? return TRUE;
}
CTest::CTest()
{
? this->data = 0;
}
void CTest::print()
{
? cout<<"The member function print() is from a DLL.\n";
}
從代碼中可以看到,DLL中定義了一個CTest類,它有一個構造函數和一個成員函數print()。
2.????? 然后新建一個Win32 Console Application來調用DLL。
Header File:
#define LIBDLL_API __declspec(dllimport)
#include <iostream.h>
// This class is exported from the LibDll.dll
class LIBDLL_API CTest
{
? int data;
public:
? CTest();
? void print();
};
Source File:
#include "stdafx.h"
#include <iostream.h>
#include <malloc.h>
#include <windows.h>
#include "LibDll.h"
typedef void (WINAPI *PCTOR)();
typedef void (*PPRINT)();
inline void CTest_print(HMODULE, CTest*);
inline void CTest_CTest(HMODULE, CTest*);
int main(int argc, char* argv[])
{
? //加載DLL
? HMODULE hmod = LoadLibrary("LibDll.dll");
? if(hmod == NULL)
? {
???????? cout<<"Failed loading DLL.\n";
???????? return 1;
? }
? //創建類對象
? CTest* pCTest = (CTest*)malloc(sizeof(CTest));
? //初始化CTest對象
? CTest_CTest(hmod, pCTest);
????????
?//調用成員函數
? CTest_print(hmod, pCTest);
? FreeLibrary(hmod);
? free(pCTest);
? cout<<"Press [Enter] to exit.";
? cin.peek();
? return 0;
}//end main
void CTest_print(HMODULE hMod, CTest* pObj)
{
? PPRINT pprint = (PPRINT)GetProcAddress(hMod, "print");
? if(pprint == NULL)
? {
???????? cout<<"Function print() not found.\n";
? }
? else
? {
???????? __asm{ MOV ECX, pObj}
???????? pprint();
? }
}
void CTest_CTest(HMODULE hMod, CTest* pObj)
{
? PCTOR pCtor = (PCTOR)GetProcAddress(hMod, "CTest");
? if(pCtor == NULL)
? {
???????? cout<<"Function CTest() not found.\n";
? }
? else
? {
???????? __asm{ MOV ECX, pObj}
???????? pCtor();
? }
}
?
本來到這里就應該可以正常運行了,但是你會發現在執行CTest_CTest函數時會提示"Function CTest() not found."。為什么會找不到函數呢?那是因為C++編譯器在生成DLL時對輸出函數的名稱進行來“修飾”,所以DLL中的函數名稱已經不再是我們在代碼中所寫的函數名,這個時候就需要用DEF文件來進行“函數名稱修復”。
3.????? 用Dumpbin的/EXPORTS參數打開LibDll.Dll,截圖如下:
?
其中1和3就是CTest類的構造函數和print()函數的實際名稱(嚇人吧……)。然后我們在DLL的工程目錄中新建一個“LibDll.def”文件,并在“工程->設置->Link”中添加參數(/def:".\libdll.def"),并編
輯DEF文件內容如下:
LIBRARY LibDll
EXPORTS
?CTest?? = ??0CTest@@QAE@XZ
?print?? = ?print@CTest@@QAEXXZ
相信你已經看出來了,這實際上是一個函數名映射。
再用Dumpbin打重新編譯得到的DLL文件,截圖如下:
圖中的4和5就是修復后的函數名。
現在再運行第二步中的程序就可以成功地調用DLL里的函數了!
尾注
文中只使用了DEF文件的一小部分功能,詳細資料請參見MSDN。
按照MSDN上的說法,有三種方法可以用來輸出函數,按推薦順序如下:
在源代碼中使用__declspec(dllexport)關鍵字(調用工程需包含*.lib)
使用DEF文件中的EXPORTS聲明(不需要*.lib,可實現動態調用)
在LINK命令中使用/EXPORT參數(效果和DEF文件相同)
具體應該使用哪種方法,還應該視用途由使用者自己決定。
?
作者水平有限,文中難免不當之處,歡迎大家指出和批評。
本文來自CSDN博客,轉載請標明出處:http://blog.csdn.net/jdcb2001/archive/2006/11/21/1401569.aspx