def文件一個比較詳細的例子
先看看EXPORTS語法規則:entryname[=internalname] [@ordinal [NONAME]] [PRIVATE] [DATA]
對上面語法的解釋:
1、
entryname 是要導出的函數名或變量名。這是必選項。如果導出的名稱與 DLL 中的名稱不同,則通過 internalname 指定 DLL 中導出的名稱。例如,如果 DLL 導出函數 func1(),要將它用作 func2(),則應指定:
EXPORTS func2=func1
2、
@ordinal 允許指定是序號而不是函數名將進入 DLL 的導出表。這有助于最小化 DLL 的大小。.LIB 文件將包含序號與函數之間的映射,這使您得以像通常在使用 DLL 的項目中那樣使用函數名。
可選的 NONAME 關鍵字允許只按序號導出,并減小結果 DLL 中導出表的大小。但是,如果要在 DLL 上使用 GetProcAddress,則必須知道序號,因為名稱將無效。
可選的 PRIVATE 關鍵字禁止將 entryname 放到由 LINK 生成的導入庫中。它對同樣是由 LINK 生成的圖像中的導出無效。
可選的 DATA 關鍵字指定導出的是數據,而不是代碼。例如,可以導出數據變量:
EXPORTS i DATA
3、當對同一導出使用 PRIVATE 和 DATA 時,PRIVATE 必須位于 DATA 的前面。
有三種導出定義的方法,按照建議的使用順序依次為:
-
源代碼中的 __declspec(dllexport) 關鍵字
-
.def 文件中的 EXPORTS 語句
-
LINK 命令中的 /EXPORT 規范
所有這三種方法可以用在同一個程序中。LINK 在生成包含導出的程序時還創建導入庫,除非生成中使用了 .exp 文件。
4、一個詳細的
EXPORTS DllCanUnloadNow @1 PRIVATE DATA DllWindowName = Name DATA DllGetClassObject @4 NONAME PRIVATE DllRegisterServer @7 DllUnregisterServer
注意,使用 .def 文件從 DLL 中導出變量時,不需要在變量上指定 __declspec(dllexport)。但是,在任何使用 DLL 的文件中,仍必須在數據聲明上使用 __declspec(dllimport)。
下面的是一個例子,可以看到def文件實際上的作用。
Node_t node;
BOOL APIENTRY DllMain( HMODULE hModule,
DWORD ul_reason_for_call,
LPVOID lpReserved
)
{
node.x = 5;
node.y = 6;
return TRUE;
}
int Max(int x, int y)
{
if(x>=y)
return x;
else
return y;
}
int Min(int x, int y)
{
if(x>=y)
return y;
else
return x;
}
Node_t * func1()
{
return &node;
}
Node_t * func2()
{
return &node;
}
下面是def文件
EXPORTS
Max =Max @2
MinChange =Min @1
func1 =func1 @3 NONAME
func2 =func2 @5 PRIVATE
node =node @8 PRIVATE
下面是對應的lib的導出
2 ?Max@@YAHHH@Z (int __cdecl Max(int,int))
1 ?MinChange@@YAHHH@Z (int __cdecl MinChange(int,int))
3 ?func1@@YAPAUNode_t@@XZ (struct Node_t * __cdecl func1(void))
下面是對應的dll文件的導出
ordinal hint RVA name
2 0 00001020 Max = ?Max@@YAHHH@Z (int __cdecl Max(int,int))
1 1 00001030 MinChange = ?Min@@YAHHH@Z (int __cdecl Min(int,int))
5 2 00001040 func2 = ?func1@@YAPAUNode_t@@XZ (struct Node_t * __cdecl func1(void))
8 3 00003348 node = ?node@@3UNode_t@@A (struct Node_t node)
3 00001040 [NONAME] ?func1@@YAPAUNode_t@@XZ (struct Node_t * __cdecl func1(void))
從上面可以看出來,def文件的符號在lib中的作用并不大,但是函數名稱,比如?MinChange@@YAHHH@Z中的MinChange是因為def中把Min改成了MinChange,所以lib中也進行了修改,但是這個MinChange符號實際上出現在dll文件的name列中。其實這也看出來了,def文件只是在loadlibrary這種運行時加載有效。
由于lib中的函數名變成了?MinChange@@YAHHH@Z,導致使用__declspec(dllimport)進行導入的啟動時加載,生成的?Min@@YAHHH@Z無法與?MinChange@@YAHHH@Z對應,而出現鏈接時的錯誤。所以不應該改變函數名,而應該在def文件中直接使用函數名,這樣啟動時加載和運行時加載都能夠順利進行。
同時從上面也可以看出NONAME和PRIVATE的作用的。
問題:試驗中使用DATA總是出錯,不知道怎么弄。