1.概論
先來闡述一下DLL(Dynamic Linkable Library)的概念,你可以簡單的把DLL看成一種倉庫,它提供給你一些可以直接拿來用的變量、函數或類。在倉庫的發展史上經歷了“無庫-靜態鏈接庫-動態鏈接庫”的時代。靜態鏈接庫與動態鏈接庫都是共享代碼的方式,如果采用靜態鏈接庫,則無論你願不願意,lib中的指令都被直接包含在最終生成的EXE文件中了。但是若使用DLL,該DLL不必被包含在最終EXE文件中,EXE文件執行時可以“動態”地引用和卸載這個與EXE獨立的DLL文件。靜態鏈接庫和動態鏈接庫的另外一個區別在於靜態鏈接庫中不能再包含其他的動態鏈接庫或者靜態庫,而在動態鏈接庫中還可以再包含其他的動態或靜態鏈接庫。
對動態鏈接庫,我們還需建立如下概念:
(1)DLL 的編制與具體的編程語言及編譯器無關
只要遵循約定的DLL接口規范和調用方式,用各種語言編寫的DLL都可以相互調用。譬如Windows提供的系統DLL(其中包括了Windows的API),在任何開發環境中都能被調用,不在乎其是Visual Basic、Visual C++還是Delphi。
(2)動態鏈接庫隨處可見
我們在Windows目錄下的system32文件夾中會看到kernel32.dll、user32.dll和gdi32.dll,Windows的大多數API都包含在這些DLL中。kernel32.dll中的函數主要處理內存管理和進程調度;user32.dll中的函數主要控制用戶界面;gdi32.dll中的函數則負責圖形方面的操作。
一般的程序員都用過類似MessageBox的函數,其實它就包含在user32.dll這個動態鏈接庫中。由此可見DLL對我們來說其實並不陌生。
(3)VC動態鏈接庫的分類
Visual C++支持三種DLL,它們分別是Non-MFC DLL(非MFC動態庫)、MFC Regular DLL(MFC規則DLL)、MFC Extension DLL(MFC擴展DLL)。
非MFC動態庫不采用MFC類庫結構,其導出函數為標准的C接口,能被非MFC或MFC編寫的應用程序所調用;MFC規則DLL 包含一個繼承自CWinApp的類,但其無消息循環;MFC擴展DLL采用MFC的動態鏈接版本創建,它只能被用MFC類庫所編寫的應用程序所調用。
由於本文篇幅較長,內容較多,勢必需要先對閱讀本文的有關事項進行說明,下面以問答形式給出。
問:本文主要講解什麼內容?
答:本文詳細介紹了DLL編程的方方面面,努力學完本文應可以對DLL有較全面的掌握,並能編寫大多數DLL程序。
問:如何看本文?
答:本文每一個主題的講解都附帶了源代碼例程,可以隨文下載(每個工程都經WINRAR壓縮)。所有這些例程都由筆者編寫並在VC++6.0中調試通過。
當然看懂本文不是讀者的最終目的,讀者應親自動手實踐才能真正掌握DLL的奧妙。
問:學習本文需要什麼樣的基礎知識?
答:如果你掌握了C,並大致掌握了C++,了解一點MFC的知識,就可以輕松地看懂本文。
2.靜態鏈接庫
對靜態鏈接庫的講解不是本文的重點,但是在具體講解DLL之前,通過一個靜態鏈接庫的例子可以快速地幫助我們建立“庫”的概念。
如圖1,在VC++6.0中new一個名稱為libTest的static library工程(單擊此處下載本工程),並新建lib.h和lib.cpp兩個文件,lib.h和lib.cpp的源代碼如下:
//文件:lib.h
#ifndef LIB_H
#define LIB_H
extern "C" int add(int x,int y); //聲明為C編譯、連接方式的外部函數
#endif
//文件:lib.cpp
#include "lib.h"
int add(int x,int y)
{
return x + y;
}
編譯這個工程就得到了一個.lib文件,這個文件就是一個函數庫,它提供了add的功能。將頭文件和.lib文件提交給用戶後,用戶就可以直接使用其中的add函數了。
標准Turbo C2.0中的C庫函數(我們用來的scanf、printf、memcpy、strcpy等)就來自這種靜態庫。
下面來看看怎麼使用這個庫,在libTest工程所在的工作區內new一個libCall工程。libCall工程僅包含一個main.cpp文件,它演示了靜態鏈接庫的調用方法,其源代碼如下:
#include <stdio.h>
#include "..lib.h"
#pragma comment( lib, "..\debug\libTest.lib" ) //指定與靜態庫一起連接
int main(int argc, char* argv[])
{
printf( "2 + 3 = %d", add( 2, 3 ) );
}
靜態鏈接庫的調用就是這麼簡單,或許我們每天都在用,可是我們沒有明白這個概念。代碼中#pragma comment( lib , "..\debug\libTest.lib" )的意思是指本文件生成的.obj文件應與libTest.lib一起連接。如果不用#pragma comment指定,則可以直接在VC++中設置,如圖2,依次選擇tools、options、directorIEs、library files菜單或選項,填入庫文件路徑。圖2中加紅圈的部分為我們添加的libTest.lib文件的路徑。
這個靜態鏈接庫的例子至少讓我們明白了庫函數是怎麼回事,它們是哪來的。我們現在有下列模糊認識了:
(1)庫不是個怪物,編寫庫的程序和編寫一般的程序區別不大,只是庫不能單獨執行;
(2)庫提供一些可以給別的程序調用的東東,別的程序要調用它必須以某種方式指明它要調用之。
以上從靜態鏈接庫分析而得到的對庫的懵懂概念可以直接引申到動態鏈接庫中,動態鏈接庫與靜態鏈接庫在編寫和調用上的不同體現在庫的外部接口定義及調用方式略有差異。
3.庫的調試與查看
在具體進入各類DLL的詳細闡述之前,有必要對庫文件的調試與查看方法進行一下介紹,因為從下一節開始我們將面對大量的例子工程。
由於庫文件不能單獨執行,因而在按下F5(開始debug模式執行)或CTRL+F5(運行)執行時,其彈出如圖3所示的對話框,要求用戶輸入可執行文件的路徑來啟動庫函數的執行。這個時候我們輸入要調用該庫的EXE文件的路徑就可以對庫進行調試了,其調試技巧與一般應用工程的調試一樣。
通常有比上述做法更好的調試途徑,那就是將庫工程和應用工程(調用庫的工程)放置在同一VC工作區,只對應用工程進行調試,在應用工程調用庫中函數的語句處設置斷點,執行後按下F11,這樣就單步進入了庫中的函數。第2節中的libTest和libCall工程就放在了同一工作區,其工程結構如圖4所示。
上述調試方法對靜態鏈接庫和動態鏈接庫而言是一致的。所以本文提供下載的所有源代碼中都包含了庫工程和調用庫的工程,這二者都被包含在一個工作區內,這是筆者提供這種打包下載的用意所在。
動態鏈接庫中的導出接口可以使用Visual C++的Depends工具進行查看,讓我們用Depends打開系統目錄中的user32.dll,看到了吧?紅圈內的就是幾個版本的MessageBox了!原來它真的在這裡啊,原來它就在這裡啊!
當然Depends工具也可以顯示DLL的層次結構,若用它打開一個可執行文件則可以看出這個可執行文件調用了哪些DLL。
好,讓我們正式進入動態鏈接庫的世界,先來看看最一般的DLL,即非MFC DLL。