DLL的進(jìn)入點(diǎn)函數(shù)
一個(gè)D L L可以擁有單個(gè)進(jìn)入點(diǎn)函數(shù)。系統(tǒng)在不同的時(shí)間調(diào)用這個(gè)進(jìn)入點(diǎn)函數(shù),這個(gè)問題將在下面加以介紹。這些調(diào)用可以用來提供一些信息,通常用于供D L L進(jìn)行每個(gè)進(jìn)程或線程的初始化和清除操作。如果你的D L L不需要這些通知信息,就不必在D L L源代碼中實(shí)現(xiàn)這個(gè)函數(shù)。例如,如果你創(chuàng)建一個(gè)只包含資源的D L L,就不必實(shí)現(xiàn)該函數(shù)。如果確實(shí)需要在D L L中接受通知信息,可以實(shí)現(xiàn)類似下面的進(jìn)入點(diǎn)函數(shù):
BOOL WINAPI DllMain(HINSTANCE hinstDll, DWORD fdwReason, PVOID fImpLoad)
{
switch(fdwReason)
{
case DLL_PROCESS_ATTACH:
//The DLL is being mapped into the process's address space.
break;
case DLL_THREAD_ATTACH:
//A thread is being created.
break;
case DLL_THREAD_DETACH:
//A thread is exiting cleanly.
break;
case DLL_PROCESS_DETACH:
//The DLL is being unmapped from the process's address space.
break;
}
return(TRUE); // Used only for DLL_PROCESS_ATTACH
}
注意函數(shù)名D l l M a i n是區(qū)分大小寫的。
參數(shù)h i n s t D l l包含了D L L的實(shí)例句柄。與( w ) Wi n M a i n函數(shù)的h i n s t E x e參數(shù)一樣,這個(gè)值用于標(biāo)識(shí)D L L的文件映像被映射到進(jìn)程的地址空間中的虛擬內(nèi)存地址。通常應(yīng)將這個(gè)參數(shù)保存在一個(gè)全局變量中,這樣就可以在調(diào)用加載資源的函數(shù)(如D i a l o g B o x和L o a d S t r i n g)時(shí)使用它。最后一個(gè)參數(shù)是f I m p L o a d,如果D L L是隱含加載的,那么該參數(shù)將是個(gè)非0值,如果D L L是顯式加載的,那么它的值是0。
參數(shù)f d w R e a s o n用于指明系統(tǒng)為什么調(diào)用該函數(shù)。該參數(shù)可以使用4個(gè)值中的一個(gè)。這4個(gè)值是: D L L _ P R O C E S S _ AT TA C H、D L L _ P R O C E S S _ D E TA C H、D L L _ T H R E A D _ AT TA C H或D L L _ T H R E A D _ D E TA C H。這些值將在下面介紹。
注意必須記住,D L L使用D l l M a i n函數(shù)來對(duì)它們進(jìn)行初始化。當(dāng)你的D l l M a i n函數(shù)執(zhí)行時(shí),同一個(gè)地址空間中的其他D L L可能尚未執(zhí)行它們的D l l M a i n函數(shù)。這意味著它們尚未初始化,因此你應(yīng)該避免調(diào)用從其他D L L中輸入的函數(shù)。此外,你應(yīng)該避免從D l l M a i n內(nèi)部調(diào)用L o a d L i b r a r y ( E x )和F r e e L i b r a r y函數(shù),因?yàn)檫@些函數(shù)會(huì)形式一個(gè)依賴性循環(huán)。
DLL_PROCESS_ATTACH通知
當(dāng)D L L被初次映射到進(jìn)程的地址空間中時(shí),系統(tǒng)將調(diào)用該D L L的D l l M a i n函數(shù),給它傳遞參數(shù)f d w R e a s o n的值D L L _ P R O C E S S _ AT TA C H。只有當(dāng)D L L的文件映像初次被映射時(shí),才會(huì)出現(xiàn)這種情況。如果線程在后來為已經(jīng)映射到進(jìn)程的地址空間中的D L L調(diào)用L o a d L i b r a r y ( E x )函數(shù),那么操作系統(tǒng)只是遞增D L L的使用計(jì)數(shù),它并不再次用D L L _ P R O C E S S _ AT TA C H的值來調(diào)用D L L的D l l M a i n函數(shù)。
當(dāng)處理D L L _ P R O C E S S _ AT TA C H時(shí),D L L應(yīng)該執(zhí)行D L L中的函數(shù)要求的任何與進(jìn)程相關(guān)的初始化。例如, D L L可能包含需要使用它們自己的堆棧(在進(jìn)程的地址空間中創(chuàng)建)的函數(shù)。
DLL_PROCESS_DETACH通知
D L L從進(jìn)程的地址空間中被卸載時(shí),系統(tǒng)將調(diào)用D L L的D l l M a i n函數(shù),給它傳遞f d w R e a s o n的值D L L _ P R O C E S S _ D E TA C H。當(dāng)D L L處理這個(gè)值時(shí),它應(yīng)該執(zhí)行任何與進(jìn)程相關(guān)的清除操作。例如, D L L可以調(diào)用H e a p D e s t r o y函數(shù)來撤消它在D L L _ P R O C E S S _ D E TA C H通知期間創(chuàng)建的堆棧。
DLL_THREAD_ATTACH通知
當(dāng)在一個(gè)進(jìn)程中創(chuàng)建線程時(shí),系統(tǒng)要查看當(dāng)前映射到該進(jìn)程的地址空間中的所有D L L文件映像,并調(diào)用每個(gè)文件映像的帶有D L L _ T H R E A D _ AT TA C H值的D l l M a i n函數(shù)。這可以告訴所有的D L L執(zhí)行每個(gè)線程的初始化操作。新創(chuàng)建的線程負(fù)責(zé)執(zhí)行D L L的所有D l l M a i n函數(shù)中的代碼。只有當(dāng)所有的D L L都有機(jī)會(huì)處理該通知時(shí),系統(tǒng)才允許新線程開始執(zhí)行它的線程函數(shù)。
DLL_THREAD_DETACH通知
讓線程終止運(yùn)行的首選方法是使它的線程函數(shù)返回。這使得系統(tǒng)可以調(diào)用E x i t T h r e a d來撤消該線程。E x i t T h r e a d函數(shù)告訴系統(tǒng),該線程想要終止運(yùn)行,但是系統(tǒng)并不立即將它撤消。相反, 它要取出這個(gè)即將被撤消的線程, 并讓它調(diào)用已經(jīng)映射的D L L 的所有帶有D L L _ T H R E A D _ D E TACH 值的D l l M a i n函數(shù)。這個(gè)通知告訴所有的D L L執(zhí)行每個(gè)線程的清除操作。
DllMain與C/C++運(yùn)行期庫
當(dāng)編寫一個(gè)D L L時(shí),你需要得到C / C + +運(yùn)行期庫的某些初始幫助。例如,如果你創(chuàng)建的D L L包含一個(gè)全局變量,而這個(gè)全局變量是個(gè)C + +類的實(shí)例。在你順利地在D l l M a i n函數(shù)中使用這個(gè)全局變量之前,該變量必須調(diào)用它的構(gòu)造函數(shù)。這是由C / C + +運(yùn)行期庫的D L L啟動(dòng)代碼來完成的。當(dāng)你的D L L文件映像被映射到進(jìn)程的地址空間中時(shí),系統(tǒng)實(shí)際上是調(diào)用_ D l l M a i n C RTS t a r t u p函數(shù),而不是調(diào)用D l l M a i n函數(shù)。
延遲加載DLL (但是怎么延遲那?^_^)
Microsoft Visual C++ 6.0提供了一個(gè)出色的新特性,它能夠使D L L的操作變得更加容易。這個(gè)特性稱為延遲加載D L L。延遲加載的D L L是個(gè)隱含鏈接的D L L,它實(shí)際上要等到你的代碼試圖引用D L L中包含的一個(gè)符號(hào)時(shí)才進(jìn)行加載。延遲加載的D L L在下列情況下是非常有用的:
? 如果你的應(yīng)用程序使用若干個(gè)D L L,那么它的初始化時(shí)間就比較長,因?yàn)榧虞d程序要將所有需要的D L L映射到進(jìn)程的地址空間中。解決這個(gè)問題的方法之一是在進(jìn)程運(yùn)行的時(shí)候分開加載各個(gè)D L L。延遲加載的D L L能夠更容易地完成這樣的加載。
? 如果調(diào)用代碼中的一個(gè)新函數(shù),然后試圖在老版本的系統(tǒng)上運(yùn)行你的應(yīng)用程序,而該系統(tǒng)中沒有該函數(shù),那么加載程序就會(huì)報(bào)告一個(gè)錯(cuò)誤,并且不允許該應(yīng)用程序運(yùn)行。你需要一種方法讓你的應(yīng)用程序運(yùn)行,然后,如果(在運(yùn)行時(shí))發(fā)現(xiàn)該應(yīng)用程序在老的系統(tǒng)上運(yùn)行,那么你將不調(diào)用遺漏的函數(shù)。
函數(shù)轉(zhuǎn)發(fā)器
函數(shù)轉(zhuǎn)發(fā)器是D L L的輸出節(jié)中的一個(gè)項(xiàng)目,用于將對(duì)一個(gè)函數(shù)的調(diào)用轉(zhuǎn)至另一個(gè)D L L中的另一個(gè)函數(shù)。
?DLL轉(zhuǎn)移
M i c r o s o f t給Windows 2000增加了一個(gè)D L L轉(zhuǎn)移特性。這個(gè)特性能夠強(qiáng)制操作系統(tǒng)的加載程序首先從你的應(yīng)用程序目錄中加載文件模塊。只有當(dāng)加載程序無法在應(yīng)用程序目錄中找到該文件時(shí),它才搜索其他目錄。為了強(qiáng)制加載程序總是首先查找應(yīng)用程序的目錄,要做的工作就是在應(yīng)用程序的目錄中放入一個(gè)文件。該文件的內(nèi)容可以忽略,但是該文件必須稱為A p p N a m e . l o c a l。例如,如果有一個(gè)可執(zhí)行文件的名字是S u p e r A p p . e x e ,那么轉(zhuǎn)移文件必須稱為S u p e r A p p . e x e . l o c a l。在系統(tǒng)內(nèi)部, L o a d L i b r a r y ( E x )已經(jīng)被修改,以便查看是否存在該文件。如果應(yīng)用程序的目錄中存在該文件,該目錄中的模塊就已經(jīng)被加載。如果應(yīng)用程序的目錄中不存在這個(gè)模塊,L o a d L i b r a r y ( E x )將正常運(yùn)行。對(duì)于已經(jīng)注冊(cè)的C O M對(duì)象來說,這個(gè)特性是非常有用的。它使應(yīng)用程序能夠?qū)⑺腃 O M對(duì)象D L L放入自己的目錄,這樣,注冊(cè)了相同C O M對(duì)象的其他應(yīng)用程序就無法干擾你的操作。
zz