DLL入門淺析(1)——如何建立DLL
初學(xué)DLL,結(jié)合教程,總結(jié)一下自己的所得,希望對(duì)DLL初學(xué)者們有所幫助。
動(dòng)態(tài)鏈接庫(DLL)是從C語言函數(shù)庫和Pascal庫單元的概念發(fā)展而來的。所有的C語言標(biāo)準(zhǔn)庫函數(shù)都存放在某一函數(shù)庫中。在鏈接應(yīng)用程序的過程中,鏈接器從庫文件中拷貝程序調(diào)用的函數(shù)代碼,并把這些函數(shù)代碼添加到可執(zhí)行文件中。這種方法同只把函數(shù)儲(chǔ)存在已編譯的OBJ文件中相比更有利于代碼的重用。但隨著Windows這樣的多任務(wù)環(huán)境的出現(xiàn),函數(shù)庫的方法顯得過于累贅。如果為了完成屏幕輸出、消息處理、內(nèi)存管理、對(duì)話框等操作,每個(gè)程序都不得不擁有自己的函數(shù),那么Windows程序?qū)⒆兊梅浅}嫶?。Windows的發(fā)展要求允許同時(shí)運(yùn)行的幾個(gè)程序共享一組函數(shù)的單一拷貝。動(dòng)態(tài)鏈接庫就是在這種情況下出現(xiàn)的。動(dòng)態(tài)鏈接庫不用重復(fù)編譯或鏈接,一旦裝入內(nèi)存,DLL函數(shù)可以被系統(tǒng)中的任何正在運(yùn)行的應(yīng)用程序軟件所使用,而不必再將DLL函數(shù)的另一拷貝裝入內(nèi)存。
下面我們一步一步來建立一個(gè)DLL。
一、建立一個(gè)DLL工程
新建一個(gè)工程,選擇Win32 控制臺(tái)項(xiàng)目(Win32 Console Application),并且在應(yīng)用程序設(shè)置標(biāo)簽(the advanced tab)上,選擇DLL和空項(xiàng)目選項(xiàng)。
二、聲明導(dǎo)出函數(shù)
這里有兩種方法聲明導(dǎo)出函數(shù):一種是通過使用__declspec(dllexport),添加到需要導(dǎo)出的函數(shù)前,進(jìn)行聲明;另外一種就是通過模塊定義文件(Module-Definition File即.DEF)來進(jìn)行聲明。
第一種方法,建立頭文件DLLSample.h,在頭文件中,對(duì)需要導(dǎo)出的函數(shù)進(jìn)行聲明。
#ifndef _DLL_SAMPLE_H
#define _DLL_SAMPLE_H

// 如果定義了C++編譯器,那么聲明為C鏈接方式
#ifdef __cplusplus

extern "C"
{
#endif

// 通過宏來控制是導(dǎo)入還是導(dǎo)出
#ifdef _DLL_SAMPLE
#define DLL_SAMPLE_API __declspec(dllexport)
#else
#define DLL_SAMPLE_API __declspec(dllimport)
#endif

// 導(dǎo)出/導(dǎo)入函數(shù)聲明
DLL_SAMPLE_API void TestDLL(int);

#undef DLL_SAMPLE_API

#ifdef __cplusplus
}
#endif

#endif
這個(gè)頭文件會(huì)分別被DLL和調(diào)用DLL的應(yīng)用程序引入,當(dāng)被DLL引入時(shí),在DLL中定義_DLL_SAMPLE宏,這樣就會(huì)在DLL模塊中聲明函數(shù)為導(dǎo)出函數(shù);當(dāng)被調(diào)用DLL的應(yīng)用程序引入時(shí),就沒有定義_DLL_SAMPLE,這樣就會(huì)聲明頭文件中的函數(shù)為從DLL中的導(dǎo)入函數(shù)。
第二種方法:模塊定義文件是一個(gè)有著.def文件擴(kuò)展名的文本文件。它被用于導(dǎo)出一個(gè)DLL的函數(shù),和__declspec(dllexport)很相似,但是.def文件并不是Microsoft定義的。一個(gè).def文件中只有兩個(gè)必需的部分:LIBRARY 和 EXPORTS。
LIBRARY DLLSample
DESCRIPTION "my simple DLL"
EXPORTS
TestDLL @1 ;@1表示這是第一個(gè)導(dǎo)出函數(shù)
第一行,''LIBRARY''是一個(gè)必需的部分。它告訴鏈接器(linker)如何命名你的DLL。下面被標(biāo)識(shí)為''DESCRIPTION''的部分并不是必需的。該語句將字符串寫入 .rdata 節(jié),它告訴人們誰可能使用這個(gè)DLL,這個(gè)DLL做什么或它為了什么(存在)。再下面的部分標(biāo)識(shí)為''EXPORTS''是另一個(gè)必需的部分;這個(gè)部分使得該函數(shù)可以被其它應(yīng)用程序訪問到并且它創(chuàng)建一個(gè)導(dǎo)入庫。當(dāng)你生成這個(gè)項(xiàng)目時(shí),不僅是一個(gè).dll文件被創(chuàng)建,而且一個(gè)文件擴(kuò)展名為.lib的導(dǎo)出庫也被創(chuàng)建了。除了前面的部分以外,這里還有其它四個(gè)部分標(biāo)識(shí)為:NAME, STACKSIZE, SECTIONS, 和 VERSION。另外,一個(gè)分號(hào)(;)開始一個(gè)注解,如同''//''在C++中一樣。定義了這個(gè)文件之后,頭文件中的__declspec(dllexport)就不需要聲明了。
三、編寫DllMain函數(shù)和導(dǎo)出函數(shù)
DllMain函數(shù)是DLL模塊的默認(rèn)入口點(diǎn)。當(dāng)Windows加載DLL模塊時(shí)調(diào)用這一函數(shù)。系統(tǒng)首先調(diào)用全局對(duì)象的構(gòu)造函數(shù),然后調(diào)用全局函數(shù)DLLMain。DLLMain函數(shù)不僅在將DLL鏈接加載到進(jìn)程時(shí)被調(diào)用,在DLL模塊與進(jìn)程分離時(shí)(以及其它時(shí)候)也被調(diào)用。
#include "stdafx.h"
#define _DLL_SAMPLE

#ifndef _DLL_SAMPLE_H
#include "DLLSample.h"
#endif

#include "stdio.h"

//APIENTRY聲明DLL函數(shù)入口點(diǎn)
BOOL APIENTRY DllMain(HANDLE 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;
}

void TestDLL(int arg)


{
printf("DLL output arg %d\n", arg);
}
如果程序員沒有為DLL模塊編寫一個(gè)DLLMain函數(shù),系統(tǒng)會(huì)從其它運(yùn)行庫中引入一個(gè)不做任何操作的缺省DLLMain函數(shù)版本。在單個(gè)線程啟動(dòng)和終止時(shí),DLLMain函數(shù)也被調(diào)用。
然后,F(xiàn)7編譯,就得到一個(gè)DLL了。
動(dòng)態(tài)鏈接庫(DLL)是從C語言函數(shù)庫和Pascal庫單元的概念發(fā)展而來的。所有的C語言標(biāo)準(zhǔn)庫函數(shù)都存放在某一函數(shù)庫中。在鏈接應(yīng)用程序的過程中,鏈接器從庫文件中拷貝程序調(diào)用的函數(shù)代碼,并把這些函數(shù)代碼添加到可執(zhí)行文件中。這種方法同只把函數(shù)儲(chǔ)存在已編譯的OBJ文件中相比更有利于代碼的重用。但隨著Windows這樣的多任務(wù)環(huán)境的出現(xiàn),函數(shù)庫的方法顯得過于累贅。如果為了完成屏幕輸出、消息處理、內(nèi)存管理、對(duì)話框等操作,每個(gè)程序都不得不擁有自己的函數(shù),那么Windows程序?qū)⒆兊梅浅}嫶?。Windows的發(fā)展要求允許同時(shí)運(yùn)行的幾個(gè)程序共享一組函數(shù)的單一拷貝。動(dòng)態(tài)鏈接庫就是在這種情況下出現(xiàn)的。動(dòng)態(tài)鏈接庫不用重復(fù)編譯或鏈接,一旦裝入內(nèi)存,DLL函數(shù)可以被系統(tǒng)中的任何正在運(yùn)行的應(yīng)用程序軟件所使用,而不必再將DLL函數(shù)的另一拷貝裝入內(nèi)存。
下面我們一步一步來建立一個(gè)DLL。
一、建立一個(gè)DLL工程
新建一個(gè)工程,選擇Win32 控制臺(tái)項(xiàng)目(Win32 Console Application),并且在應(yīng)用程序設(shè)置標(biāo)簽(the advanced tab)上,選擇DLL和空項(xiàng)目選項(xiàng)。
二、聲明導(dǎo)出函數(shù)
這里有兩種方法聲明導(dǎo)出函數(shù):一種是通過使用__declspec(dllexport),添加到需要導(dǎo)出的函數(shù)前,進(jìn)行聲明;另外一種就是通過模塊定義文件(Module-Definition File即.DEF)來進(jìn)行聲明。
第一種方法,建立頭文件DLLSample.h,在頭文件中,對(duì)需要導(dǎo)出的函數(shù)進(jìn)行聲明。




























這個(gè)頭文件會(huì)分別被DLL和調(diào)用DLL的應(yīng)用程序引入,當(dāng)被DLL引入時(shí),在DLL中定義_DLL_SAMPLE宏,這樣就會(huì)在DLL模塊中聲明函數(shù)為導(dǎo)出函數(shù);當(dāng)被調(diào)用DLL的應(yīng)用程序引入時(shí),就沒有定義_DLL_SAMPLE,這樣就會(huì)聲明頭文件中的函數(shù)為從DLL中的導(dǎo)入函數(shù)。
第二種方法:模塊定義文件是一個(gè)有著.def文件擴(kuò)展名的文本文件。它被用于導(dǎo)出一個(gè)DLL的函數(shù),和__declspec(dllexport)很相似,但是.def文件并不是Microsoft定義的。一個(gè).def文件中只有兩個(gè)必需的部分:LIBRARY 和 EXPORTS。





三、編寫DllMain函數(shù)和導(dǎo)出函數(shù)
DllMain函數(shù)是DLL模塊的默認(rèn)入口點(diǎn)。當(dāng)Windows加載DLL模塊時(shí)調(diào)用這一函數(shù)。系統(tǒng)首先調(diào)用全局對(duì)象的構(gòu)造函數(shù),然后調(diào)用全局函數(shù)DLLMain。DLLMain函數(shù)不僅在將DLL鏈接加載到進(jìn)程時(shí)被調(diào)用,在DLL模塊與進(jìn)程分離時(shí)(以及其它時(shí)候)也被調(diào)用。

































然后,F(xiàn)7編譯,就得到一個(gè)DLL了。
posted on 2009-07-20 13:49 Saga 閱讀(23034) 評(píng)論(2) 編輯 收藏 引用 所屬分類: Windows