[DLL - Beginers]使用DEF文件修復(fù)函數(shù)名——對《使用LoadLibrary調(diào)用從Dll中輸出的class》的一點(diǎn)補(bǔ)充 收藏
使用DEF文件修復(fù)函數(shù)名
——對《使用LoadLibrary調(diào)用從Dll中輸出的class》的一點(diǎn)補(bǔ)充
作者 李成竹
在《使用……》一文中,作者在“代碼”的第三點(diǎn)提到了“使用一個(gè)DEF文件修復(fù)了函數(shù)名”,但是并沒有講解什么是DEF文件,也沒有說明應(yīng)該如何修復(fù),可能會使某些初學(xué)者(包括我自己)感到疑惑。我也上網(wǎng)搜索了一下,講解DEF文件作用以及詳細(xì)使用方法的文章不多且比較零散,本文在此用一個(gè)簡單例子簡單闡述一下DEF文件一般的使用方法,以方便需要者查閱。
?
DEF文件的全稱是Module-Definition File,即模塊定義文件,是用來定義EXE和DLL文件的一種文件格式,以文本形式保存(可用記事本創(chuàng)建/編輯)。由于鏈接器為大多數(shù)模塊定義聲明提供了對應(yīng)的命令行選項(xiàng),所以一般的Win32程序并不需要.DEF文件。但是在編寫DLL時(shí),尤其是在編寫C++的DLL時(shí),(由于名稱修飾)DEF文件還是有它的用武之地的。
※注:關(guān)于“名稱修飾”在很多地方都有介紹,文中不作講解。
?
DEF文件的主要內(nèi)容是由一系列的聲明(statement)組成,包括NAME、LIBRARY、DISCRIPTION、STACKSIZE、SECTIONS、EXPORTS、VERSION。
?
¨???????? NAME:指定輸出文件的文件名,設(shè)置image基址
¨???????? LIBRARY(DLL):指定DLL的內(nèi)部名稱和加載時(shí)的基址
¨???????? DISCRIPTION:文件描述
¨???????? STACKSIZE:設(shè)置棧的大小
¨???????? SECTIONS:設(shè)置image文件的一個(gè)或多個(gè)段屬性
¨???????? EXPORTS:定義輸出列表
¨???????? VERSION:指定文件版本
?
其中最常用的是LIBRARY、EXPORTS和DISCRIPTION。
?
示例
1.????? 現(xiàn)在有一個(gè)已經(jīng)編寫好的類要用DLL輸出,并通過函數(shù)名對DLL進(jìn)行動態(tài)調(diào)用。其頭文件和源文件如下:
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中定義了一個(gè)CTest類,它有一個(gè)構(gòu)造函數(shù)和一個(gè)成員函數(shù)print()。
2.????? 然后新建一個(gè)Win32 Console Application來調(diào)用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;
? }
? //創(chuàng)建類對象
? CTest* pCTest = (CTest*)malloc(sizeof(CTest));
? //初始化CTest對象
? CTest_CTest(hmod, pCTest);
????????
?//調(diào)用成員函數(shù)
? 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();
? }
}
?
本來到這里就應(yīng)該可以正常運(yùn)行了,但是你會發(fā)現(xiàn)在執(zhí)行CTest_CTest函數(shù)時(shí)會提示"Function CTest() not found."。為什么會找不到函數(shù)呢?那是因?yàn)镃++編譯器在生成DLL時(shí)對輸出函數(shù)的名稱進(jìn)行來“修飾”,所以DLL中的函數(shù)名稱已經(jīng)不再是我們在代碼中所寫的函數(shù)名,這個(gè)時(shí)候就需要用DEF文件來進(jìn)行“函數(shù)名稱修復(fù)”。
3.????? 用Dumpbin的/EXPORTS參數(shù)打開LibDll.Dll,截圖如下:
?
其中1和3就是CTest類的構(gòu)造函數(shù)和print()函數(shù)的實(shí)際名稱(嚇人吧……)。然后我們在DLL的工程目錄中新建一個(gè)“LibDll.def”文件,并在“工程->設(shè)置->Link”中添加參數(shù)(/def:".\libdll.def"),并編
輯DEF文件內(nèi)容如下:
LIBRARY LibDll
EXPORTS
?CTest?? = ??0CTest@@QAE@XZ
?print?? = ?print@CTest@@QAEXXZ
相信你已經(jīng)看出來了,這實(shí)際上是一個(gè)函數(shù)名映射。
再用Dumpbin打重新編譯得到的DLL文件,截圖如下:
圖中的4和5就是修復(fù)后的函數(shù)名。
現(xiàn)在再運(yùn)行第二步中的程序就可以成功地調(diào)用DLL里的函數(shù)了!
尾注
文中只使用了DEF文件的一小部分功能,詳細(xì)資料請參見MSDN。
按照MSDN上的說法,有三種方法可以用來輸出函數(shù),按推薦順序如下:
在源代碼中使用__declspec(dllexport)關(guān)鍵字(調(diào)用工程需包含*.lib)
使用DEF文件中的EXPORTS聲明(不需要*.lib,可實(shí)現(xiàn)動態(tài)調(diào)用)
在LINK命令中使用/EXPORT參數(shù)(效果和DEF文件相同)
具體應(yīng)該使用哪種方法,還應(yīng)該視用途由使用者自己決定。
?
作者水平有限,文中難免不當(dāng)之處,歡迎大家指出和批評。
本文來自CSDN博客,轉(zhuǎn)載請標(biāo)明出處:http://blog.csdn.net/jdcb2001/archive/2006/11/21/1401569.aspx