? 在工作中經(jīng)常碰到動(dòng)態(tài)鏈接庫(kù),一直以來都沒有好好去理解它,這幾天潛下心來仔細(xì)的研究了一下它,在這里把我看到一些東西的和實(shí)驗(yàn)得出的一些心得整理一下。
引言
? 動(dòng)態(tài)鏈接庫(kù)(DLL)是WINDOWS平臺(tái)的基礎(chǔ),Windows API 中的所有函數(shù)都包含在D L L中。有3個(gè)最重要的D L L是K e r n e l 3 2 . d l l,它包含用于管理內(nèi)存、進(jìn)程和線程的各個(gè)函數(shù); U s e r 3 2 . d l l,它包含用于執(zhí)行用戶界面任務(wù)(如窗口的創(chuàng)建和消息的傳送)的各個(gè)函數(shù); G D I 3 2 . d l l,它包含用于畫圖和顯示文本的各個(gè)函數(shù)。
它有以下幾個(gè)優(yōu)點(diǎn):
→節(jié)約內(nèi)存;
→使應(yīng)用程序“變瘦”;
→可單獨(dú)修改動(dòng)態(tài)鏈接庫(kù)而不必與應(yīng)用程序重新鏈接;
→可方便實(shí)現(xiàn)多語(yǔ)言聯(lián)合編程(比如用VC++寫個(gè)dll,然后在VB中調(diào)用);
→可將資源打包;
→可在應(yīng)用程序間共享內(nèi)存
→可以用于一些特殊的目的。例如安裝鉤子函數(shù)
?
關(guān)于擴(kuò)展名
? 一般來說,DLL是一種磁盤文件,以.dll、.DRV、.FON、.SYS和許多以.EXE為擴(kuò)展名的系統(tǒng)文件都可以是DLL , 但只有擴(kuò)展名為dll的動(dòng)態(tài)鏈接庫(kù)才能被Windows自動(dòng)載入。如果使用其它擴(kuò)展名的動(dòng)態(tài)鏈接庫(kù),則調(diào)用動(dòng)態(tài)鏈接庫(kù)的程序中必須使用LoadLibrary或LoadLibraryEx載入動(dòng)態(tài)鏈接庫(kù)模塊。
關(guān)于調(diào)用方式
? 隱式的調(diào)用:需要把產(chǎn)生動(dòng)態(tài)連接庫(kù)時(shí)產(chǎn)生的.LIB文件加入到應(yīng)用程序的工程中,想使用DLL中的函數(shù)時(shí),只須說明一下。隱式調(diào)用不需要調(diào)用LoadLibrary()和FreeLibrary()。當(dāng)你建立一個(gè)DLL文件時(shí),鏈接程序會(huì)自動(dòng)生成一個(gè)與之對(duì)應(yīng)的LIB導(dǎo)入文件。該文件包含了每一個(gè)DLL導(dǎo)出函數(shù)的符號(hào)名和可選的標(biāo)識(shí)號(hào),但是并不含有實(shí)際的代碼。LIB文件作為DLL的替代文件被編譯到應(yīng)用程序項(xiàng)目中。當(dāng)你通過靜態(tài)鏈接方式編譯生成應(yīng)用程序時(shí),應(yīng)用程序中的調(diào)用函數(shù)與LIB文件中導(dǎo)出符號(hào)相匹配,這些符號(hào)或標(biāo)識(shí)號(hào)進(jìn)入到生成的EXE文件中。LIB文件中也包含了對(duì)應(yīng)的DLL文件名(但不是完全的路徑名),鏈接程序?qū)⑵浯鎯?chǔ)在EXE文件內(nèi)部。當(dāng)應(yīng)用程序運(yùn)行過程中需要加載DLL文件時(shí),Windows根據(jù)這些信息發(fā)現(xiàn)并加載DLL,然后通過符號(hào)名或標(biāo)識(shí)號(hào)實(shí)現(xiàn)對(duì)DLL函數(shù)的動(dòng)態(tài)鏈接。所有被應(yīng)用程序調(diào)用的DLL文件都會(huì)在應(yīng)用程序EXE文件加載時(shí)被加載在到內(nèi)存中。可執(zhí)行程序鏈接到一個(gè)包含DLL輸出函數(shù)信息的輸入庫(kù)文件(.LIB文件)。操作系統(tǒng)在加載使用可執(zhí)行程序時(shí)加載DLL。可執(zhí)行程序直接通過函數(shù)名調(diào)用DLL的輸出函數(shù),調(diào)用方法和程序內(nèi)部其他的函數(shù)是一樣的。
? 顯式的調(diào)用:是指在應(yīng)用程序中用LoadLibrary或MFC提供的AfxLoadLibrary顯式的將自己所做的動(dòng)態(tài)連接庫(kù)調(diào)進(jìn)來,動(dòng)態(tài)連接庫(kù)的文件名即是上面兩個(gè)函數(shù)的參數(shù),再用GetProcAddress()獲取想要引入的函數(shù)。自此,你就可以象使用如同本應(yīng)用程序自定義的函數(shù)一樣來調(diào)用此引入函數(shù)了。在應(yīng)用程序退出之前,應(yīng)該用FreeLibrary或MFC提供的AfxFreeLibrary釋放動(dòng)態(tài)連接庫(kù)。直接調(diào)用Win32 的LoadLibary函數(shù),并指定DLL的路徑作為參數(shù)。LoadLibary返回HINSTANCE參數(shù),應(yīng)用程序在調(diào)用GetProcAddress函數(shù)時(shí)使用這一參數(shù)。GetProcAddress函數(shù)將符號(hào)名或標(biāo)識(shí)號(hào)轉(zhuǎn)換為DLL內(nèi)部的地址。程序員可以決定DLL文件何時(shí)加載或不加載,顯式鏈接在運(yùn)行時(shí)決定加載哪個(gè)DLL文件。使用DLL的程序在使用之前必須加載(LoadLibrary)加載DLL從而得到一個(gè)DLL模塊的句柄,然后調(diào)用GetProcAddress函數(shù)得到輸出函數(shù)的指針,在退出之前必須卸載DLL(FreeLibrary)。
Windows將遵循下面的搜索順序來定位DLL:
包含EXE文件的目錄
進(jìn)程的當(dāng)前工作目錄
Windows系統(tǒng)目錄
Windows目錄
列在Path環(huán)境變量中的一系列目錄
關(guān)于生成和使用DLL
?? 以VS2005為例,建立一個(gè)WIN32工程,工程名為Try,應(yīng)用程序類型選擇“DLL”,附加選項(xiàng)選擇“導(dǎo)出符號(hào)”。編譯器會(huì)自動(dòng)的生成Try.h和Try.cpp文件。





















































































??
? 在頭文件的前面使用_ _ d e c l s p e c ( d l l e x p o r t )對(duì)M Y L I B A P I進(jìn)行定義。當(dāng)編譯器看到負(fù)責(zé)修改變量、函數(shù)或C + +類的_ _ d e c l s p e c ( d l l e x p o r t )時(shí),它就知道該變量、函數(shù)或C + +類是從產(chǎn)生的D L L模塊輸出的。
仔細(xì)分析一下這段代碼:
#ifdefTRY_EXPORTS??
#defineTRY_API__declspec(dllexport)
#else
#define TRY_API __declspec(dllimport)
#endif
//如果已定義了TRY_EXPORTS
//用TRY_API替換__declspec(dllexport)
//否則
//將TRY_API替換__declspec(dllimport)
??當(dāng)編譯器看到負(fù)責(zé)修飾變量、函數(shù)或C + +類的_ _ d e c l s p e c ( d l l e x p o r t )時(shí),它就知道該變量、函數(shù)或C + +類是從產(chǎn)生的D L L模塊輸出的,當(dāng)看到_ d e c l s p e c ( d l l im p o r t )時(shí),就知道它是輸入的。
? 那么TRY_EXPORTS有沒有定義呢?在哪里定義的?在項(xiàng)目屬性—>c/c++—>預(yù)處理器中有這么一行,預(yù)處理器定義:WIN32;_DEBUG;_WINDOWS;_USRDLL;TRY_EXPORTS。
? 建一個(gè)測(cè)試工程,將生成的Try.dll,Try.lib,Try.pdb統(tǒng)統(tǒng)copy到該工程目錄下,在工程屬性—>鏈接器—>輸入中將Try.lib添加進(jìn)去。就可以像普通工程那樣使用DLL中的定義的數(shù)據(jù)了。
















