今天在研究怎么在vc中調(diào)用動(dòng)態(tài)dll的問題,看了一個(gè)下午,總算有些眉目。
首先來(lái)說說
調(diào)用的原理:
調(diào)用DLL,首先需要將DLL文件映像到用戶進(jìn)程的地址空間中,然后才能進(jìn)行函數(shù)調(diào)用,這個(gè)函數(shù)和進(jìn)程內(nèi)部一般函數(shù)的調(diào)用方法相同。Windows提供了
兩種將DLL映像到進(jìn)程地址空間的方法:隱式調(diào)用(通過lib和頭文件)和顯式調(diào)用(只通過提供的dll文件)。下面對(duì)這兩種方式在vc中如何調(diào)用做詳細(xì)的說明:
a.隱式:
這種方法需要DLL工程經(jīng)編譯產(chǎn)生的LIB文件,此文件中包含了DLL允許應(yīng)用程序調(diào)用的所有函數(shù)的列表,當(dāng)鏈接器發(fā)現(xiàn)應(yīng)用程序調(diào)用了LIB文件列出的某
個(gè)函數(shù),就會(huì)在應(yīng)用程序的可執(zhí)行文件的文件映像中加入一些信息,這些信息指出了包含這個(gè)函數(shù)的DLL文件的名字。當(dāng)這個(gè)應(yīng)用程序運(yùn)行時(shí),也就是它的可執(zhí)行
文件被操作系統(tǒng)產(chǎn)生映像文件時(shí),系統(tǒng)會(huì)查看這個(gè)映像文件中關(guān)于DLL的信息,然后將這個(gè)DLL文件映像到進(jìn)程的地址空間。
系統(tǒng)通過DLL文件的名稱,試圖加載這個(gè)文件到進(jìn)程地址空間時(shí),它尋找DLL
文件的路徑按照先后順序如下:
·程序運(yùn)行時(shí)的目錄,即可執(zhí)行文件所在的目錄;
·當(dāng)前程序工作目錄
·系統(tǒng)目錄:對(duì)于Windows95/98來(lái)說,可以調(diào)用GetSystemDirectory函數(shù)來(lái)得到,對(duì)于WindowsNT/2000來(lái)說,指的
是32位Windows的系統(tǒng)目錄,也可以調(diào)用GetSystemDirectory函數(shù)來(lái)得到,得到的值為SYSTEM32。
·Windows目錄
·列在PATH環(huán)境變量中的所有目錄
VC中加載DLL的LIB文件的方法有以下三種:
①LIB文件直接加入到工程文件列表中
在VC中打開File
View一頁(yè),選中工程名,單擊鼠標(biāo)右鍵,然后選中“Add Files to
Project”菜單,在彈出的文件對(duì)話框中選中要加入DLL的LIB文件即可。
②設(shè)置工程的 Project Settings來(lái)加載DLL的LIB文件
打開工程的 Project
Settings菜單,選中Link,然后在Object/library
modules下的文本框中輸入DLL的LIB文件。
③通過程序代碼的方式
加入預(yù)編譯指令#pragma comment
(lib,”*.lib”),這種方法優(yōu)點(diǎn)是可以利用條件預(yù)編譯指令鏈接不同版本的LIB文件。因?yàn)?,在Debug方式下,產(chǎn)生的LIB文件是Debug版本,如Regd.lib;在Release方式下,產(chǎn)生的LIB文件是Release版本,如Regr.lib。
當(dāng)應(yīng)用程序?qū)LL的LIB文件加載后,還需要把DLL對(duì)應(yīng)的頭文件(*.h)包含到其中,在這個(gè)頭文件中給出了DLL中定義的函數(shù)原型,然后聲明。
b.顯式
隱式鏈接雖然實(shí)現(xiàn)較簡(jiǎn)單,但除了必須的*.dll文件外還需要DLL的*.h文件和*.lib文件,在那些只提供*.dll文件的場(chǎng)合就無(wú)法使用,而只能
采用顯式鏈接的方式。這種方式通過調(diào)用API函數(shù)來(lái)完成對(duì)DLL的加載與卸載,其能更加有效地使用內(nèi)存,在編寫大型應(yīng)用程序時(shí)往往采用此方式。這種方法編
程具體實(shí)現(xiàn)步驟如下:
①使用Windows API函數(shù)Load
Library或者M(jìn)FC提供的AfxLoadLibrary將DLL模塊映像到進(jìn)程的內(nèi)存空間,對(duì)DLL模塊進(jìn)行動(dòng)態(tài)加載。
②使用GetProcAddress函數(shù)得到要調(diào)用DLL中的函數(shù)的指針。
③不用DLL時(shí),用Free
Library函數(shù)或者AfxFreeLibrary函數(shù)從進(jìn)程的地址空間顯式卸載DLL
VC中調(diào)用實(shí)例
數(shù)據(jù)加密是計(jì)算機(jī)安全領(lǐng)域的重要內(nèi)容,其基本思想是通過變換信息的表現(xiàn)形式來(lái)保護(hù)敏感信息,使非授權(quán)者不能了解被保護(hù)信息的內(nèi)容[4]。常見的數(shù)據(jù)加密算法有:DES,IDEA,RSA,ECC,AES,MD5,SHA等。
《共享軟件加密算法庫(kù)》是一款針對(duì)個(gè)人、企業(yè)開發(fā)共享軟件的加密工具,支持Windows平臺(tái)下各類開發(fā)工具:VC、VB、Delphi、PB、VFP
等,算法庫(kù)集成的算法有:BlowFish、MD5、Secret16、AES、SHA、CRC32、RSA、DES、字符串加/解密、文件加/解密等多
種功能強(qiáng)大的算法。其提供了DLL文件—Reg.dll,可以通過復(fù)用它來(lái)實(shí)現(xiàn)數(shù)據(jù)加密與解密。
隱式鏈接
其提供了
Reg.h與Reg.lib兩個(gè)隱式鏈接所必須的文件,所以可以采用此種方式。
①在VC中打開File
View一頁(yè),選中工程名,單擊鼠標(biāo)右鍵,然后選中“Add Files to
Project”菜單,在彈出的文件對(duì)話框中選中要加入Reg.lib。
②在VC中打開File View一頁(yè),選中Header
files,單擊鼠標(biāo)右鍵,然后選中“Add Files to
Folder”菜單,在彈出的文件對(duì)話框中選中要加入Reg.h,然后在工程相應(yīng)的頭文件中加入#i
nclude
"Reg.h"。在Reh.h頭文件中給出了DLL中定義的函數(shù)原型及聲明。
如:加密函數(shù)原型及聲明為extern "C" BOOL WINAPI File
Encrypt(LPCTSTR lpInputFileName, LPCTSTR lpOutputFileName, LPCTSTR
lpKey, LPCTSTR lpRegisterCode);解密函數(shù)原型及聲明為extern "C" BOOL
WINAPI File Decrypt(LPCTSTR lpInputFileName, LPCTSTR
lpOutputFileName, LPCTSTR lpKey, LPCTSTR
lpRegisterCode)。其中對(duì)于WINAPI宏,把它加到函數(shù)原型定義前,系統(tǒng)會(huì)把它翻譯為適當(dāng)?shù)恼{(diào)用方式,在Win32中,是把它翻譯為_stdcall調(diào)用方式。
③直接調(diào)用所需要的加密與解密函數(shù),如調(diào)用File
Encrypt()函數(shù)實(shí)現(xiàn)文本文件和二進(jìn)制文件的加密,調(diào)用File
Decrypt()函數(shù)實(shí)現(xiàn)文本文件和二進(jìn)制文件的解密,調(diào)用時(shí)的參數(shù)要與函數(shù)定義參數(shù)相符合。
顯式鏈接
如果只提供Reg.dll一個(gè)文件,那么須用此種方式。
①加密模塊:調(diào)用File
Encrypt()函數(shù)實(shí)現(xiàn)文本文件和二進(jìn)制文件的加密。
//裝載加密/解密DLL
HINSTANCE hdll=::Load Library ("Reg.dll");
//通過類型定義語(yǔ)句typedef來(lái)定義函數(shù)指針類型
Typedef BOOL (_stdcall
*lpFileEncrypt)(LPCTSTR, LPCTSTR, LPCTSTR, LPCTSTR);
//函數(shù)聲明
LpFileEncrypt FileEncrypt1;
//獲取加密函數(shù)File Encrypt的函數(shù)指針
FileEncrypt1=(lpFileEncrypt)::GetProcAddress(hdll,"FileEncrypt");
//調(diào)用DLL中加密函數(shù)File
Encrypt對(duì)文件加密,user-12345678為軟件注冊(cè)號(hào)
FileEncrypt1(加密源文件名,加密生成目標(biāo)文件名,密碼,"user-12345678");
//釋放DLL模塊
::AfxFreeLibrary(hdll);
②解密模塊:調(diào)用File
Decrypt()函數(shù)實(shí)現(xiàn)文本文件和二進(jìn)制文件的解密。
//裝載加密/解密DLL
HINSTANCE hdll=::Load Library ("Reg.dll");
//通過類型定義語(yǔ)句typedef來(lái)定義函數(shù)指針類型
Typedef BOOL (_stdcall *lpFileDecrypt)(LPCTSTR,
LPCTSTR, LPCTSTR, LPCTSTR);
//函數(shù)聲明
LpFileDecrypt FileDecrypt2;
//獲取解密函數(shù)File Decrypt的函數(shù)指針
FileDecrypt2=(lpFileDecrypt)::GetProcAddress(hdll,"FileDecrypt");
//調(diào)用DLL中解密函數(shù)FileDecrypt對(duì)文件加密,user-12345678為軟件注冊(cè)號(hào)
FileDecrypt2(解密源文件名,解密生成目標(biāo)文件名,密碼,"user-12345678");
//釋放DLL模塊
::AfxFreeLibrary(hdll);
附microsoft visual Stdio
MSDN關(guān)于dll顯式調(diào)用的相關(guān)說明
// File:
RUNTIME.C
// A simple program that uses LoadLibrary and
// GetProcAddress to access myPuts from MYPUTS.DLL.
#include
typedef VOID (*MYPROC)(LPTSTR);
VOID main(VOID)
{ HINSTANCE hinstLib;
MYPROC ProcAdd;
BOOL fFreeResult,
fRunTimeLinkSuccess = FALSE;
// Get a handle
to the DLL
module.
hinstLib =
LoadLibrary(“myputs“);
// If the handle is valid, try to get the function
address.
if (hinstLib != NULL)
{ ProcAdd = (MYPROC) GetProcAddress(hinstLib,
“myPuts“);
// If the function address is valid, call the function.
if (NULL != ProcAdd) { fRunTimeLinkSuccess = TRUE;
(ProcAdd) (“message via DLL function\n“);
} // Free the DLL module. fFreeResult =
FreeLibrary(hinstLib);
} // If unable to call the DLL function, use an
alternative.
if (! fRunTimeLinkSuccess)
printf(“message via alternative method\n“);
}