?
? 在工作中經(jīng)常碰到動(dòng)態(tài)鏈接庫,一直以來都沒有好好去理解它,這幾天潛下心來仔細(xì)的研究了一下它,在這里把我看到一些東西的和實(shí)驗(yàn)得出的一些心得整理一下。
引言
? 動(dòng)態(tài)鏈接庫(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)鏈接庫而不必與應(yīng)用程序重新鏈接;
→可方便實(shí)現(xiàn)多語言聯(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)鏈接庫才能被Windows自動(dòng)載入。如果使用其它擴(kuò)展名的動(dòng)態(tài)鏈接庫,則調(diào)用動(dòng)態(tài)鏈接庫的程序中必須使用LoadLibrary或LoadLibraryEx載入動(dòng)態(tài)鏈接庫模塊。
?
關(guān)于調(diào)用方式
? 隱式的調(diào)用:需要把產(chǎn)生動(dòng)態(tài)連接庫時(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ù)信息的輸入庫文件(.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)連接庫調(diào)進(jìn)來,動(dòng)態(tài)連接庫的文件名即是上面兩個(gè)函數(shù)的參數(shù),再用GetProcAddress()獲取想要引入的函數(shù)。自此,你就可以象使用如同本應(yīng)用程序自定義的函數(shù)一樣來調(diào)用此引入函數(shù)了。在應(yīng)用程序退出之前,應(yīng)該用FreeLibrary或MFC提供的AfxFreeLibrary釋放動(dòng)態(tài)連接庫。直接調(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文件。
//Try.h
#ifdef?TRY_EXPORTS
#define?TRY_API?__declspec(dllexport)
#else
#define?TRY_API?__declspec(dllimport)
#endif
#include?<iostream>
using?namespace?std;

//?此類是從?Try.dll?導(dǎo)出的

class?TRY_API?CTry?
{
public:
????CTry(void)?;
????//?TODO:?在此添加您的方法。

????void?print()?;
????
};
extern?TRY_API?int?nTry;

TRY_API?int?fnTry(void);
//?Try.cpp?:?定義?DLL?應(yīng)用程序的入口點(diǎn)。
//

#include?"stdafx.h"
#include?"Try.h"


#ifdef?_MANAGED
#pragma?managed(push,?off)
#endif

BOOL?APIENTRY?DllMain(?HMODULE?hModule,
???????????????????????DWORD??ul_reason_for_call,
???????????????????????LPVOID?lpReserved
?????????????????????)


{
????switch?(ul_reason_for_call)

????
{
????case?DLL_PROCESS_ATTACH:
????case?DLL_THREAD_ATTACH:
????case?DLL_THREAD_DETACH:
????case?DLL_PROCESS_DETACH:
????????break;
????}
????return?TRUE;
}

#ifdef?_MANAGED
#pragma?managed(pop)
#endif

//?這是導(dǎo)出變量的一個(gè)示例
TRY_API?int?nTry?=?0;

//?這是導(dǎo)出函數(shù)的一個(gè)示例。
TRY_API?int?fnTry(void)


{
????return?42;
}

//?這是已導(dǎo)出類的構(gòu)造函數(shù)。
//?有關(guān)類定義的信息,請(qǐng)參閱?Try.h
CTry::CTry()


{
????return??;
}

void?CTry::print()


{
????cout<<"CTry"<<endl;
}

??
? 在頭文件的前面使用_ _ 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ù)了。
#include?"stdafx.h"
#include?<iostream>
#include?"Try.h"
using?namespace?std;

int?main()


{
????cout<<nTry<<endl;
????cout<<fnTry()<<endl;
????CTry?_a;
????_a.print()?;
????system("pause")?;
????
????return?0?;
}
posted on 2009-03-06 00:59
隙中駒 閱讀(484)
評(píng)論(1) 編輯 收藏 引用