DLL的建立與調(diào)用
[轉(zhuǎn)]
動(dòng)態(tài)鏈接庫(kù)是一個(gè)能夠被應(yīng)用程序和其它的DLL調(diào)用的過程和函數(shù)的集合體,它里面包含的是公共代碼或資源。由于DLL代碼使用了內(nèi)存共享技術(shù),在某些地方windows也給了DLL一些更高的權(quán)限,因而DLL中可以實(shí)現(xiàn)一些一般程序所不能實(shí)現(xiàn)的功能,如實(shí)現(xiàn)windows的HOOK、ISAPI等。
同時(shí),DLL還為不同語言間代碼共享提供了一條方便的途徑。因而DLL在編程時(shí)應(yīng)用較為廣泛,本文將介紹如何在 Delphi 中建立和使用DLL。
一.DLL 庫(kù)內(nèi)存共享機(jī)制
從使用效果看,DLL和unit 很像,它們都可以被別的工程模塊所調(diào)用,但二者在內(nèi)部的實(shí)現(xiàn)機(jī)制上確存在著差別。如果一個(gè)程序模塊中用uses語句引用了某個(gè)unit,編譯程序在編譯該模塊時(shí),便會(huì)連同unit一起編譯,并把編譯后的可執(zhí)行代碼鏈接到本程序模塊中,這就是一個(gè)程序模塊能夠調(diào)用所引用unit中過程和函數(shù)的原因。
當(dāng)同一個(gè)unit被多個(gè)工程所引用時(shí),則每個(gè)工程中都含有該unit的可執(zhí)行代碼,當(dāng)含有該unit的多個(gè)工程同時(shí)執(zhí)行時(shí),unit的可執(zhí)行代碼會(huì)隨不同工程而多次被調(diào)入內(nèi)存,造成內(nèi)存資源的浪費(fèi)。DLL則不同,它即使被某個(gè)工程調(diào)用,編譯后仍是獨(dú)立的。
也就是說編譯后,一個(gè)DLL庫(kù)形成一個(gè)單獨(dú)的可執(zhí)行文件,而不與任何其它的可執(zhí)行文件連接在一起,因而DLL庫(kù)并不從屬于某個(gè)特定的工程,當(dāng)多個(gè)工程調(diào)用同一個(gè)DLL庫(kù)時(shí)只有第一個(gè)工程把DLL庫(kù)調(diào)入內(nèi)存,其余工程并不重復(fù)調(diào)入同一個(gè)DLL庫(kù)到內(nèi)存,而是到同一個(gè)共享內(nèi)存區(qū)讀取。并且,DLL的執(zhí)行代碼是在程序運(yùn)行期間動(dòng)態(tài)調(diào)入的,而不是如unit在程序運(yùn)行時(shí)就與整個(gè)工程一起調(diào)入內(nèi)存。這樣便可消除unit帶來的相同代碼多處占用內(nèi)存的弊病。
二 Delphi中DLL庫(kù)的建立
在Delphi環(huán)境中,編寫一個(gè)DLL同編寫一個(gè)一般的應(yīng)用程序并沒有太大的區(qū)別。事實(shí)上作為DLL主體的DLL函數(shù)的編寫,除了在內(nèi)存、資源的管理上有所不同外,并不需要其它特別的手段。
一般工程文件的格式為:
program 工程標(biāo)題;
uses 子句;
程序體
而DLLs工程文件的格式為:
library 工程標(biāo)題;
uses 子句;
exprots 子句;
絳蛺?
它們主要的區(qū)別有兩點(diǎn):
1.一般工程文件的頭標(biāo)用program關(guān)鍵字,而DLL工程文件頭標(biāo)用library 關(guān)鍵字。不同的關(guān)鍵字通知編譯器生成不同的可執(zhí)行文件。用program關(guān)鍵字生成的是.exe文件,而用library關(guān)鍵字生成的是.dll文件;
2.假如DLL要輸出供其它應(yīng)用程序使用的函數(shù)或過程,則必須將這些函數(shù)或過程列在exports子句中。而這些函數(shù)或過程本身必須用export編譯指令進(jìn)行編譯。 在Delphi主菜單file 中選new...項(xiàng),在彈出的窗口中雙擊DLL圖標(biāo),便會(huì)自動(dòng)給出DLL源模塊框架,如下:
Library project1;
{...注釋...}
uses
SysUtils, Classes;
begin
end.
接下來便可在USES和begin之間加入想在該DLL中實(shí)現(xiàn)的過程和函數(shù)的定義,并用export和exprots保字把它們引出,以便別的模塊引用,在begin和end之間加入初始化代碼,初始化代碼是用來對(duì)DLL變量初始化的。應(yīng)注意,即便無初始化代碼begin與end也不可省略,如下例:
library minmax;
function Min(X, Y: Integer): Integer; export;
begin
if X < Y then Min := X else Min := Y;
end;
function Max(X, Y: Integer): Integer; export;
begin
if X > Y then Max := X else Max := Y;
end;
exports
Min index 1,
Max index 2;
begin
end.
經(jīng)編譯后,并以minmax.DLL存盤后,一個(gè)DLL庫(kù)文件便形成了。
三 DLL庫(kù)的訪問
訪問DLL庫(kù)有兩種方式,一種是靜態(tài)引用,另一種是動(dòng)態(tài)引用。
用靜態(tài)引用這種方法裝入DLL要做兩件事情:為DLL 庫(kù)創(chuàng)建一個(gè)輸入單元,以及用USES把輸入單元連接到要使用DLL 函數(shù)的程序模塊中。為DLL庫(kù)創(chuàng)建的輸入單元與普通的單元的區(qū)別僅在于:在它的接口處聲明的過程、函數(shù),并不在它的實(shí)現(xiàn)部分給出真正的實(shí)現(xiàn)代碼,而是用external關(guān)鍵字把過程、函數(shù)的實(shí)現(xiàn)細(xì)節(jié)委托給外部DLL模塊。
external命令的使用語法如下:
procedure /function 過程/函數(shù)名;external DLL模塊名;
下面給出為上面創(chuàng)建的minmax.DLL庫(kù)寫的輸入單元源文件testdll .pas,從中可看出輸入單元與一般單元的一些差別,代碼如下所示:
unit testdll;
interface
uses
function Min (X, Y: Integer): Integer;
function Max (X, Y: Integer): Integer;
implementation
function Min; external ‘minmax.DLL’;
function Max; external ‘minmax.DLL’;
end.
一個(gè)應(yīng)用程序若想調(diào)用minmax.DLL中的函數(shù),只須在其uses語句中加入testdll 單元即可。
動(dòng)態(tài)裝入DLL,要用到Windows的三個(gè)API函數(shù)。Loadlibrary、Freelibrary和GetprocAddress 。 loadlibrary函數(shù)用來裝入DLL庫(kù),其調(diào)用格式如下:
function loadlobrary (DLLfileName:Pchar): THandle:
當(dāng)不再需要一個(gè)DLL庫(kù)時(shí),應(yīng)調(diào)用FreeLibrary函數(shù)將其釋放,以空出寶貴的內(nèi)存資源,其調(diào)用格式如下:
procedure FreeLibrary (Libmodule:THandle)
Libmodule 為由LoadLibrary調(diào)用得到的DLL庫(kù)句柄。在用loadlobrary 函數(shù)裝入某個(gè)DLL庫(kù)和調(diào)用FreeLibrary釋放該DLL庫(kù)之間的程序段中, 可以使用該DLL庫(kù)中的過程和函數(shù),具體使用方法是:用GetprocAddress函數(shù)把DLL庫(kù)中函數(shù)的地址傳遞給程序中某個(gè)函數(shù)變量,再用該變量實(shí)現(xiàn)DLL函數(shù)的調(diào)用。GetprocAddress函數(shù)聲名如下
function GetprocAddress (Libmodule:THandle:procname:pchar):TFarProc:
如下例所示:
type
TTimeRec = record
Second: Integer;
Minute: Integer;
Hour: Integer;
end;
TGetTime = procedure(var Time: TTimeRec);
THandle = Integer;
var
Time: TTimeRec;
Handle: THandle;
GetTime: TGetTime;
...
begin
Handle := LoadLibrary('DATETIME.DLL');
if Handle <> 0 then
begin
@GetTime := GetProcAddress(Handle, 'GetTime');
if @GetTime <> nil then
begin
GetTime(Time);
with Time do
WriteLn('The time is ', Hour, ':', Minute, ':', Second);
end;
FreeLibrary(Handle);
end;
end;
在調(diào)用動(dòng)態(tài)鏈接庫(kù)時(shí)應(yīng)注意, 所需動(dòng)態(tài)鏈接庫(kù)須與應(yīng)用程序在同一目錄或Windows System 目錄下。
動(dòng)態(tài)鏈接庫(kù)是 Windows下程序組織的一種重要方式,使用動(dòng)態(tài)鏈接庫(kù)可以極大地保護(hù)用戶在不同開發(fā)工具、不同時(shí)期所做的工作,提高編程效率。
一 Dll的制作一般步驟
二 參數(shù)傳遞
三 DLL的初始化和退出清理[如果需要初始化和退出清理]
四 全局變量的使用
五 調(diào)用靜態(tài)載入
六 調(diào)用動(dòng)態(tài)載入
七 在DLL建立一個(gè)TForM
八 在DLL中建立一個(gè)TMDIChildForM
九 示例:
十 Delphi制作的Dll與其他語言的混合編程中常遇問題:
十一 相關(guān)資料
一 Dll的制作一般分為以下幾步:
1 .在一個(gè)DLL工程里寫一個(gè)過程或函數(shù)
2 .寫一個(gè)Exports關(guān)鍵字,在其下寫過程的名稱。不用寫參數(shù)和調(diào)用后綴。
二 參數(shù)傳遞
1 .參數(shù)類型最好與window C++的參數(shù)類型一致。不要用DELPHI的數(shù)據(jù)類型。
2 .最好有返回值[即使是一個(gè)過程],來報(bào)出調(diào)用成功或失敗,或狀態(tài)。成功或失敗的返回值最好為1[成功]或0[失敗].一句話,與windows c++兼容。
3 .用stdcall聲明后綴。
4 .最好大小寫敏感。
5 .無須用far調(diào)用后綴,那只是為了與windows 16位程序兼容。
三 DLL的初始化和退出清理[如果需要初始化和退出清理]
1 .DLLProc[SysUtils單元的一個(gè)Pointer]是DLL的入口。在此你可用你的函數(shù)替換了它的入口。但你的函數(shù)必須符合以下要求[其實(shí)就是一個(gè)回調(diào)函數(shù)]。如下:
procedure DllEnterPoint(dwReason: DWORD);far;stdcall;
dwReason參數(shù)有四種類型:
DLL_PROCESS_ATTACH:進(jìn)程進(jìn)入時(shí)
DLL_PROCESS_DETACH進(jìn)程退出時(shí)
DLL_THREAD_ATTACH 線程進(jìn)入時(shí)
DLL_THREAD_DETACH 線程退出時(shí)
在初始化部分寫:
DLLProc := @DLLEnterPoint;
DllEnterPoint(DLL_PROCESS_ATTACH);
2 .如Form上有TdcomConnection組件,就Uses Activex,在初始化時(shí)寫一句CoInitialize (nil);
3 .在退出時(shí)一定保證DcomConnection.Connected := False,并且數(shù)據(jù)集已關(guān)閉。否則報(bào)地址錯(cuò)。
四 全局變量的使用
在widnows 32位程序中,兩個(gè)應(yīng)用程序的地址空間是相互沒有聯(lián)系的。雖然DLL在內(nèi)存中是一份,但變量是在各進(jìn)程的地址空間中,因此你不能借助dll的全局變量來達(dá)到兩個(gè)應(yīng)用程序間的數(shù)據(jù)傳遞,除非你用內(nèi)存映像文件。
五 調(diào)用靜態(tài)載入
1 客戶端函數(shù)聲名:
1)大小寫敏感。
)與DLL中的聲明一樣。
如: showform(form:Tform);Far;external'yproject_dll.dll';
3)調(diào)用時(shí)傳過去的參數(shù)類型最好也與windows c++一樣。
4)調(diào)用時(shí)DLL必須在windows搜索路徑中,順序是:當(dāng)前目錄;Path路徑;windows;widows\system;windows\ssystem32;
六 調(diào)用動(dòng)態(tài)載入
1 .建立一種過程類型[如果你對(duì)過程類型的變量只是一個(gè)指針的本質(zhì)清楚的話,你就知道是怎么回事了]。如:
type
mypointer=procedure(form:Tform);Far;external;
var
Hinst:Thandle;
showform:mypointer;
begin
Hinst:=loadlibrary('yproject_dll');//Load一個(gè)Dll,按文件名找。
showform:=getprocaddress(Hinst,'showform');//按函數(shù)名找,大小寫敏感。如果你知道自動(dòng)化對(duì)象的本質(zhì)就清楚了。
showform(application.mainform);//找到函數(shù)入口指針就調(diào)用。
Freelibrary(Hinst);
end;
七 .在DLL建立一個(gè)TForM
1 把你的Form Uses到Dll中,你的Form用到的關(guān)聯(lián)的單元也要Uses進(jìn)來[這是最麻煩的一點(diǎn),因?yàn)槟愕腇orm或許Uses了許多特殊的單元或函數(shù)]
2 傳遞一個(gè)Application參數(shù),用它建立Form.
八 .在DLL中建立一個(gè)TMDIChildForM
1 Dll中的MDIForm.FormStyle不用為fmMDIChild.
2 在CreateForm后寫以下兩句:
function ShowForm(mainForm:TForm):integer;stdcall
var
Form1: TForm1;
ptr:PLongInt;
begin
ptr:=@(Application.MainForm);//先把dll的MainForm句柄保存起來,也無須釋放,只不過是替換一下
ptr^:=LongInt(mainForm);//用主調(diào)程序的mainForm替換DLL的MainForm。MainForm是特殊的WINDOW,它專門管理Application中的Forms資源.
//為什么不直接Application.MainForm := mainForm,因?yàn)锳pplication.MainForm是只讀屬性
Form1:=TForm1.Create(mainForm);//用參數(shù)建立
end;
備注:參數(shù)是主調(diào)程序的Application.MainForm
九 .示例:
DLL源代碼:
library Project2;
uses
SysUtils,
Classes,
Dialogs,
Forms,
Unit2 in 'Unit2.pas' {Form2};
{$R *.RES}
var
ccc: Pchar;
procedure OpenForm(mainForm:TForm);stdcall;
var
Form1: TForm1;
ptr:PLongInt;
begin
ptr:=@(Application.MainForm);
ptr^:=LongInt(mainForm);
Form1:=TForm1.Create(mainForm);
end;
procedure InputCCC(Text: Pchar);stdcall;
begin
ccc := Text;
end;
procedure ShowCCC;stdcall;
begin
ShowMessage(String(ccc));
end;
exports
OpenForm;
InputCCC,
ShowCCC;
begin
end.
調(diào)用方源代碼:
unit Unit1;
interface
uses
Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs,
StdCtrls;
type
TForm1 = class(TForm)
Button1: TButton;
Button2: TButton;
Edit1: TEdit;
procedure Button1Click(Sender: TObject);
procedure Button2Click(Sender: TObject);
private
{ Private declarations }
public
{ Public declarations }
end;
var
Form1: TForm1;
implementation
{$R *.DFM}
procedure OpenForm(mainForm:TForm);stdcall;External'project2.dll';
procedure ShowCCC;stdcall;External'project2.dll';
procedure InputCCC(Text: Pchar);stdcall;External'project2.dll';
procedure TForm1.Button1Click(Sender: TObject);
var
Text: Pchar;
begin
Text := Pchar(Edit1.Text);
// OpenForm(Application.MainForm);//為了調(diào)MDICHILD
InputCCC(Text);//為了實(shí)驗(yàn)DLL中的全局變量是否在各個(gè)應(yīng)用程序間共享
end;
procedure TForm1.Button2Click(Sender: TObject);
begin
ShowCCC;//這里表明WINDOWS 32位應(yīng)用程序DLL中的全局變量也是在應(yīng)用程序地址空間中,16位應(yīng)用程序或許不同,沒有做實(shí)驗(yàn)。
end;
十 Delphi制作的Dll與其他語言的混合編程中常遇問題:
1 .與PowerBuilder混合編程
在定義不定長(zhǎng)動(dòng)態(tài)數(shù)組方面在函數(shù)退出清理堆棧時(shí)老出現(xiàn)不可重現(xiàn)的地址錯(cuò),原因未明,大概與PB的編譯器原理有關(guān),即使PB編譯成二進(jìn)制代碼也如此。