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