DLL入門淺析(1)——如何建立DLL
初學(xué)DLL,結(jié)合教程,總結(jié)一下自己的所得,希望對DLL初學(xué)者們有所幫助。
動態(tài)鏈接庫(DLL)是從C語言函數(shù)庫和Pascal庫單元的概念發(fā)展而來的。所有的C語言標準庫函數(shù)都存放在某一函數(shù)庫中。在鏈接應(yīng)用程序的過程中,鏈接器從庫文件中拷貝程序調(diào)用的函數(shù)代碼,并把這些函數(shù)代碼添加到可執(zhí)行文件中。這種方法同只把函數(shù)儲存在已編譯的OBJ文件中相比更有利于代碼的重用。但隨著Windows這樣的多任務(wù)環(huán)境的出現(xiàn),函數(shù)庫的方法顯得過于累贅。如果為了完成屏幕輸出、消息處理、內(nèi)存管理、對話框等操作,每個程序都不得不擁有自己的函數(shù),那么Windows程序?qū)⒆兊梅浅}嫶蟆indows的發(fā)展要求允許同時運行的幾個程序共享一組函數(shù)的單一拷貝。動態(tài)鏈接庫就是在這種情況下出現(xiàn)的。動態(tài)鏈接庫不用重復(fù)編譯或鏈接,一旦裝入內(nèi)存,DLL函數(shù)可以被系統(tǒng)中的任何正在運行的應(yīng)用程序軟件所使用,而不必再將DLL函數(shù)的另一拷貝裝入內(nèi)存。
下面我們一步一步來建立一個DLL。
一、建立一個DLL工程
新建一個工程,選擇Win32 控制臺項目(Win32 Console Application),并且在應(yīng)用程序設(shè)置標簽(the advanced tab)上,選擇DLL和空項目選項。
二、聲明導(dǎo)出函數(shù)
這里有兩種方法聲明導(dǎo)出函數(shù):一種是通過使用__declspec(dllexport),添加到需要導(dǎo)出的函數(shù)前,進行聲明;另外一種就是通過模塊定義文件(Module-Definition File即.DEF)來進行聲明。
第一種方法,建立頭文件DLLSample.h,在頭文件中,對需要導(dǎo)出的函數(shù)進行聲明。
#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
這個頭文件會分別被DLL和調(diào)用DLL的應(yīng)用程序引入,當(dāng)被DLL引入時,在DLL中定義_DLL_SAMPLE宏,這樣就會在DLL模塊中聲明函數(shù)為導(dǎo)出函數(shù);當(dāng)被調(diào)用DLL的應(yīng)用程序引入時,就沒有定義_DLL_SAMPLE,這樣就會聲明頭文件中的函數(shù)為從DLL中的導(dǎo)入函數(shù)。
第二種方法:模塊定義文件是一個有著.def文件擴展名的文本文件。它被用于導(dǎo)出一個DLL的函數(shù),和__declspec(dllexport)很相似,但是.def文件并不是Microsoft定義的。一個.def文件中只有兩個必需的部分:LIBRARY 和 EXPORTS。
LIBRARY DLLSample
DESCRIPTION "my simple DLL"
EXPORTS
TestDLL @1 ;@1表示這是第一個導(dǎo)出函數(shù)
第一行,''LIBRARY''是一個必需的部分。它告訴鏈接器(linker)如何命名你的DLL。下面被標識為''DESCRIPTION''的部分并不是必需的。該語句將字符串寫入 .rdata 節(jié),它告訴人們誰可能使用這個DLL,這個DLL做什么或它為了什么(存在)。再下面的部分標識為''EXPORTS''是另一個必需的部分;這個部分使得該函數(shù)可以被其它應(yīng)用程序訪問到并且它創(chuàng)建一個導(dǎo)入庫。當(dāng)你生成這個項目時,不僅是一個.dll文件被創(chuàng)建,而且一個文件擴展名為.lib的導(dǎo)出庫也被創(chuàng)建了。除了前面的部分以外,這里還有其它四個部分標識為:NAME, STACKSIZE, SECTIONS, 和 VERSION。另外,一個分號(;)開始一個注解,如同''//''在C++中一樣。定義了這個文件之后,頭文件中的__declspec(dllexport)就不需要聲明了。
三、編寫DllMain函數(shù)和導(dǎo)出函數(shù)
DllMain函數(shù)是DLL模塊的默認入口點。當(dāng)Windows加載DLL模塊時調(diào)用這一函數(shù)。系統(tǒng)首先調(diào)用全局對象的構(gòu)造函數(shù),然后調(diào)用全局函數(shù)DLLMain。DLLMain函數(shù)不僅在將DLL鏈接加載到進程時被調(diào)用,在DLL模塊與進程分離時(以及其它時候)也被調(diào)用。
#include "stdafx.h"
#define _DLL_SAMPLE

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

#include "stdio.h"

//APIENTRY聲明DLL函數(shù)入口點
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模塊編寫一個DLLMain函數(shù),系統(tǒng)會從其它運行庫中引入一個不做任何操作的缺省DLLMain函數(shù)版本。在單個線程啟動和終止時,DLLMain函數(shù)也被調(diào)用。
然后,F(xiàn)7編譯,就得到一個DLL了。
動態(tài)鏈接庫(DLL)是從C語言函數(shù)庫和Pascal庫單元的概念發(fā)展而來的。所有的C語言標準庫函數(shù)都存放在某一函數(shù)庫中。在鏈接應(yīng)用程序的過程中,鏈接器從庫文件中拷貝程序調(diào)用的函數(shù)代碼,并把這些函數(shù)代碼添加到可執(zhí)行文件中。這種方法同只把函數(shù)儲存在已編譯的OBJ文件中相比更有利于代碼的重用。但隨著Windows這樣的多任務(wù)環(huán)境的出現(xiàn),函數(shù)庫的方法顯得過于累贅。如果為了完成屏幕輸出、消息處理、內(nèi)存管理、對話框等操作,每個程序都不得不擁有自己的函數(shù),那么Windows程序?qū)⒆兊梅浅}嫶蟆indows的發(fā)展要求允許同時運行的幾個程序共享一組函數(shù)的單一拷貝。動態(tài)鏈接庫就是在這種情況下出現(xiàn)的。動態(tài)鏈接庫不用重復(fù)編譯或鏈接,一旦裝入內(nèi)存,DLL函數(shù)可以被系統(tǒng)中的任何正在運行的應(yīng)用程序軟件所使用,而不必再將DLL函數(shù)的另一拷貝裝入內(nèi)存。
下面我們一步一步來建立一個DLL。
一、建立一個DLL工程
新建一個工程,選擇Win32 控制臺項目(Win32 Console Application),并且在應(yīng)用程序設(shè)置標簽(the advanced tab)上,選擇DLL和空項目選項。
二、聲明導(dǎo)出函數(shù)
這里有兩種方法聲明導(dǎo)出函數(shù):一種是通過使用__declspec(dllexport),添加到需要導(dǎo)出的函數(shù)前,進行聲明;另外一種就是通過模塊定義文件(Module-Definition File即.DEF)來進行聲明。
第一種方法,建立頭文件DLLSample.h,在頭文件中,對需要導(dǎo)出的函數(shù)進行聲明。
#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
這個頭文件會分別被DLL和調(diào)用DLL的應(yīng)用程序引入,當(dāng)被DLL引入時,在DLL中定義_DLL_SAMPLE宏,這樣就會在DLL模塊中聲明函數(shù)為導(dǎo)出函數(shù);當(dāng)被調(diào)用DLL的應(yīng)用程序引入時,就沒有定義_DLL_SAMPLE,這樣就會聲明頭文件中的函數(shù)為從DLL中的導(dǎo)入函數(shù)。
第二種方法:模塊定義文件是一個有著.def文件擴展名的文本文件。它被用于導(dǎo)出一個DLL的函數(shù),和__declspec(dllexport)很相似,但是.def文件并不是Microsoft定義的。一個.def文件中只有兩個必需的部分:LIBRARY 和 EXPORTS。
LIBRARY DLLSample
DESCRIPTION "my simple DLL"
EXPORTS
TestDLL @1 ;@1表示這是第一個導(dǎo)出函數(shù)
三、編寫DllMain函數(shù)和導(dǎo)出函數(shù)
DllMain函數(shù)是DLL模塊的默認入口點。當(dāng)Windows加載DLL模塊時調(diào)用這一函數(shù)。系統(tǒng)首先調(diào)用全局對象的構(gòu)造函數(shù),然后調(diào)用全局函數(shù)DLLMain。DLLMain函數(shù)不僅在將DLL鏈接加載到進程時被調(diào)用,在DLL模塊與進程分離時(以及其它時候)也被調(diào)用。
#include "stdafx.h"
#define _DLL_SAMPLE
#ifndef _DLL_SAMPLE_H
#include "DLLSample.h"
#endif
#include "stdio.h"
//APIENTRY聲明DLL函數(shù)入口點
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);
}然后,F(xiàn)7編譯,就得到一個DLL了。
posted on 2009-07-20 13:49 Saga 閱讀(23053) 評論(2) 編輯 收藏 引用 所屬分類: Windows

