對自己所做的事要有興趣,同時還要能夠堅持不懈
entryname 是要導(dǎo)出的函數(shù)名或變量名。這是必選項。如果導(dǎo)出的名稱與 DLL 中的名稱不同,則通過 internalname 指定 DLL 中導(dǎo)出的名稱。例如,如果 DLL 導(dǎo)出函數(shù) func1(),要將它用作 func2(),則應(yīng)指定:
EXPORTS func2=func1
@ordinal 允許指定是序號而不是函數(shù)名將進(jìn)入 DLL 的導(dǎo)出表。這有助于最小化 DLL 的大小。.LIB 文件將包含序號與函數(shù)之間的映射,這使您得以像通常在使用 DLL 的項目中那樣使用函數(shù)名。
可選的 NONAME 關(guān)鍵字允許只按序號導(dǎo)出,并減小結(jié)果 DLL 中導(dǎo)出表的大小。但是,如果要在 DLL 上使用 GetProcAddress,則必須知道序號,因為名稱將無效。
可選的 PRIVATE 關(guān)鍵字禁止將 entryname 放到由 LINK 生成的導(dǎo)入庫中。它對同樣是由 LINK 生成的圖像中的導(dǎo)出無效。
可選的 DATA 關(guān)鍵字指定導(dǎo)出的是數(shù)據(jù),而不是代碼。例如,可以導(dǎo)出數(shù)據(jù)變量:
EXPORTS i DATA
當(dāng)對同一導(dǎo)出使用 PRIVATE 和 DATA 時,PRIVATE 必須位于 DATA 的前面。
有三種導(dǎo)出定義的方法,按照建議的使用順序依次為:
源代碼中的 __declspec(dllexport) 關(guān)鍵字
.def 文件中的 EXPORTS 語句
LINK 命令中的 /EXPORT 規(guī)范
所有這三種方法可以用在同一個程序中。LINK 在生成包含導(dǎo)出的程序時還創(chuàng)建導(dǎo)入庫,除非生成中使用了 .exp 文件。
EXPORTS DllCanUnloadNow @1 PRIVATE DATA DllWindowName = Name DATA DllGetClassObject @4 NONAME PRIVATE DllRegisterServer @7 DllUnregisterServer
注意,使用 .def 文件從 DLL 中導(dǎo)出變量時,不需要在變量上指定 __declspec(dllexport)。但是,在任何使用 DLL 的文件中,仍必須在數(shù)據(jù)聲明上使用 __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文件 EXPORTSMax =Max @2MinChange =Min @1func1 =func1 @3 NONAME func2 =func2 @5 PRIVATE node =node @8 PRIVATE 下面是對應(yīng)的lib的導(dǎo)出 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)) 下面是對應(yīng)的dll文件的導(dǎo)出 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中的作用并不大,但是函數(shù)名稱,比如?MinChange@@YAHHH@Z中的MinChange是因為def中把Min改成了MinChange,所以lib中也進(jìn)行了修改,但是這個MinChange符號實際上出現(xiàn)在dll文件的name列中。其實這也看出來了,def文件只是在loadlibrary這種運行時加載有效。由于lib中的函數(shù)名變成了?MinChange@@YAHHH@Z,導(dǎo)致使用__declspec(dllimport)進(jìn)行導(dǎo)入的啟動時加載,生成的?Min@@YAHHH@Z無法與?MinChange@@YAHHH@Z對應(yīng),而出現(xiàn)鏈接時的錯誤。所以不應(yīng)該改變函數(shù)名,而應(yīng)該在def文件中直接使用函數(shù)名,這樣啟動時加載和運行時加載都能夠順利進(jìn)行。同時從上面也可以看出NONAME和PRIVATE的作用的。問題:試驗中使用DATA總是出錯,不知道怎么弄。
下面是def文件
下面是對應(yīng)的lib的導(dǎo)出
下面是對應(yīng)的dll文件的導(dǎo)出
從上面可以看出來,def文件的符號在lib中的作用并不大,但是函數(shù)名稱,比如?MinChange@@YAHHH@Z中的MinChange是因為def中把Min改成了MinChange,所以lib中也進(jìn)行了修改,但是這個MinChange符號實際上出現(xiàn)在dll文件的name列中。其實這也看出來了,def文件只是在loadlibrary這種運行時加載有效。由于lib中的函數(shù)名變成了?MinChange@@YAHHH@Z,導(dǎo)致使用__declspec(dllimport)進(jìn)行導(dǎo)入的啟動時加載,生成的?Min@@YAHHH@Z無法與?MinChange@@YAHHH@Z對應(yīng),而出現(xiàn)鏈接時的錯誤。所以不應(yīng)該改變函數(shù)名,而應(yīng)該在def文件中直接使用函數(shù)名,這樣啟動時加載和運行時加載都能夠順利進(jìn)行。同時從上面也可以看出NONAME和PRIVATE的作用的。問題:試驗中使用DATA總是出錯,不知道怎么弄。
posted on 2009-08-24 10:29 把握命運 閱讀(15750) 評論(1) 編輯 收藏 引用
我比較愚鈍,還是看不出PRIVATE設(shè)置的作用是什么 兄臺如果方便的話QQ聯(lián)系我下,十分感激 另外,DATA設(shè)置 只能用在導(dǎo)出變量上,不能用在函數(shù)上 (QQ:29654761) 回復(fù) 更多評論
Powered by: C++博客 Copyright © 把握命運