Delphi 编写DLL动态链接库文件的知识
一、DLL动态链接库文件的知识简介:
Windows的发展要求允许同时运行的几个程序共享一组函数的单一拷贝。动态链接库就是在这种情况下出现的。动态链接库不用重复编译或链接,一旦装入内存,Dlls函数可以被系统中的任何正在运行的应用程序软件所使用,而不必再将DLLs函数的另一拷贝装入内存。 任何应用程序都可以共享由装入内存的DLLs管理的内存资源块。只包含共享数据的DLLs称为资源文件。在Delphi中,一般工程文件的头标用program关键字,而DLLs工程文件头标用library 关键字标识(ActiveX控件也是一样)。不同的关键字通知编译器生成不同的可执行文件。用program关键字生成的是.exe文件,而用library关键字生成的是.dll等其他文件;假如要输出供其它应用程序使用的函数或过程,则必须将这些函数或过程列在Exports子句中。而这些函数或过程本身必须用export编译指令进行编译。、
使用DLL动态链接库技术主要有以下几个原因:
1>、减少可执行文件的大小;
2>、实现资源共享;
3>、便于维护和升级
4>、比较安全
二、DLL动态链接库文件的分类:
根据DLLs完成的功能,我们把DLLs分为如下的三类:
1、完成一般功能的DLLs;
2、用于数据交换的DLLs;
3、用于窗体重用的DLLs。
三、DLL动态链接库文件的基本格式如下:
library Project1; // 定义DLL文件的文件名,也是库名。和Unit差不多,会随保存时的文件名一起改变
uses
SysUtils,
Classes,
Unit1 in 'Unit1.pas' {Form1}, // 创建的窗体文件
Unit2 in 'Unit2.pas'; // 创建的单元文件
Type
// 定义自己的数据类型
Var
// 定义变量。
// 自己定义的函数
function TestDll(i:integer):integer;stdcall; // 与平时的编写差不多,只是多了一个stdcall参数
begin
Result := i+i;
end;
{$R *.res} // 设置版本信息Project|options,必须有{$R *.res}才能显示。也可以位于函数的定义之前。
// 自己定义的函数
exports // 将函数或过程输出,供其他程序使用。不用写参数和调用后缀。函数直接用‘,‘分开;
TestDll;
begin
end.
四、创建和调用DLL动态链接库的基本步骤:
1、点击【File】—>【New】—>【Other】菜单项,打开【New Items】,选择【New】;
2、选择【Dll Wizard】选项卡,点击ok,DLL工程创建成功。
3、添加代码。
4、按【Project】的【Build Project1】生成DLL动态链接库文件Project1.DLL。
5、调用DLL动态链接库文件。
//调用程序和Project1.dll在同一个目录中,在implementation下面写, external后指定了Delphi.dll的位置
1>、function TestDll(i:integer):integer;stdcall; external ‘Project1.dll’;
//TestDll 必须跟Dll中函数名一样,区分大小写;Project1不区分大小写;
2>、使用就跟普通的函数是一样的。
五、编写DLL动态链接库时,应该注意的事项:
1、在DLL中编写的函数或过程都必须加上stdcall调用参数。
在Delphi 1或Delphi 2环境下该调用参数是far。从Delphi 3以后将这个参数变为了stdcall,目的是为了使用标准的Win32参数传递技术来代替优化的register参数。忘记使用stdcall参数是常见的错误,这个错误不会影响DLL的编译和生成,但当调用这个DLL时会发生很严重的错误,导致操作系统的死锁。原因是register参数是Delphi的默认参
数。如果确实,就会变成register了。
2、所写的函数和过程应该用exports语句声明为外部函数。
正如大家看到的,TestDll函数被声明为一个外部函数。这样做可以使该函数在外部就能看到,具体方法是单激鼠标右键用“快速查看(Quick View)”功能查看该DLL文件。(如果没有“快速查看”选项可以从Windows CD上安装。)TestDll函数会出现在Export Table栏中。另一个很充分的理由是,如果不这样声明,我们如果不这样声明,我们编写的函数将不能被调用,这是大家都不愿看到的。
3、当使用了长字符串类型的参数、变量时要引用ShareMem,或者避免使用String类型。
Delphi中的string类型很强大,我们知道普通的字符串长度最大为256个字符,但Delphi中string类型在默认情况下长度可以达到2G。(对,您没有看错,确实是两兆。)这时,如果您坚持要使用string类型的参数、变量甚至是记录信息时,就要引用ShareMem单元,而且必须是第一个引用的。既在uses语句后是第一个引用的单元。如下例:uses ShareMem, SysUtils, Classes;
还有一点,在您的工程文件(*.dpr)中而不是单元文件(*.pas)中也要做同样的工作,这一点Delphi自带的帮助文件没有说清楚,造成了很多误会。不这样做的话,您很有可能付出死机的代价? 避免使用string类型的方法是将string类型的参数、变量等声明为Pchar或ShortString(如:s:string[10])类型。同样的问题会出现在当您使用了动态数组时,解决的方法同上所述。
六、在Delphi中调用DLL:
在Delphi中调用DLL动态链接库有两种方法:静态调用方法、动态调用方法;
1、静态调用DLL动态链接库(如上面给出的格式一样)
unit Unit1;
interface
uses
Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs, StdCtrls;
type
TForm1 = class(TForm)
Edit1: TEdit; // 编辑框(Edit)
Button1: TButton; // 按钮(Button)
procedure Button1Click(Sender: TObject);
private
{ Private declarations }
public
{ Public declarations }
end;
var
Form1: TForm1;
implementation
{$R *.DFM}
// 本行以下代码为我们真正动手写的代码
function TestDll(i:integer):integer;stdcall; external ‘Project1.dll ';
procedure TForm1.Button1Click(Sender: TObject);
begin
Edit1.Text:=IntToStr(TestDll(1));
end;
end.
注意事项有以下一些:
1>、调用参数用stdcall。
和前面提到的一样,当引用DLL中的函数和过程时也要使用stdcall参数,原因和前面提到的一样。
2>、用external语句指定被调用的DLL文件的路径和名称。
正如大家看到的,我们在external语句中指定了所要调用的DLL文件的名称。没有写路径是因为该DLL文件和调用它的主程序在同一目录下。如果DLL文件在C:\,则我们可将上面的引用语句写为external 'C:\Delphi.dll '。注意文件的后缀.dll必须写上。
3>、不能从DLL中调用全局变量。
如果我们在DLL中声明了某种全局变量,如:var s:byte。这样在DLL中s这个全局变量是可以正常使用的,但s不能被调用程序使用,既s不能作为全局变量传递给调用程序。不过在调用程序中声明的变量可以作为参数传递给DLL。
4>、被调用的DLL必须存在。
这一点很重要,使用静态调用方法时要求所调用的DLL文件以及要调用的函数或过程等等必须存在。如果不存在或指定的路径和文件名不正确的话,运行主程序时系统会提示“启动程序时出错”或“找不到*.dll文件”等运行错误。
2、动态调用DLL动态链接库
只是将原来的Button1Click过程中的语句用下面的代码替换掉了。
procedure TForm1.Button1Click(Sender: TObject);
type
TIntFunc=function(i:integer):integer;stdcall; //定义一个函数类型
var
Th: Thandle;
Tf: TIntFunc;
Tp: TFarProc;
begin
Th := LoadLibrary( ‘Project1.dll'); // 装载DLL文件
if Th>0 then
try
Tp:=GetProcAddress(Th,PChar(‘TestDll’)); // 查找函数的位置
if Tp<>nil then
begin
Tf := TIntFunc(Tp);
Edit1.Text := IntToStr(Tf(1)); // 调用TestC函数
end
else
ShowMessage(‘TestDll函数没有找到 ');
Finally
FreeLibrary(Th); // 释放DLL,否则会一直占用内存,知道退出windows或关机为止;
End
else
ShowMessage( 'Project1.dll没有找到 ');
end;
大家已经看到了,这种动态调用技术很复杂,但只要修改参数,如修改LoadLibrary( 'Project1.dll ')中的DLL名称为'Delphi.dll '就可动态更改所调用的DLL。
注意的事项有以下:
1>、定义所要调用的函数或过程的类型。
在上面的代码中我们定义了一个TIntFunc类型,这是对应我们将要调用的函数TestDll的。在其他调用情况下也要做同样的定义工作。并且也要加上stdcall调用参数。
2>、释放所调用的DLL。
我们用LoadLibrary动态的调用了一个DLL,但要记住必须在使用完后手动地用FreeLibrary将该DLL释放掉,否则该DLL将一直占用内存直到您退出Windows或关机为止。
3、两种调用方法之间的优缺点:
静态方法实现简单,易于掌握并且一般来说稍微快一点,也更加安全可靠一些;但是静态方法不能灵活地在运行时装卸所需的DLL,而是在主程序开始运行时就装载指定的DLL直到程序结束时才释放该DLL,另外只有基于编译器和链接器的系统(如Delphi)才可以使用该方法。
动态方法较好地解决了静态方法中存在的不足,可以方便地访问DLL中的函数和过程,甚至一些老版本DLL中新添加的函数或过程;但动态方法难以完全掌握,使用时因为不同的函数或过程要定义很多很复杂的类型和调用方法。对于初学者,笔者建议您使用静态方法,待熟练后再使用动态调用方法。
七、使用DLL的实用技巧:
1、编写技巧:
1>、为了保证DLL的正确性,可先编写成普通的应用程序的一部分,调试无误后再从主程序中分离出来,编译成DLL。
2>、为了保证DLL的通用性,应该在自己编写的DLL中杜绝出现可视化控件的名称,如:Edit1.Text中的Edit1名称;或者自
定义非Windows定义的类型,如某种记录。
3>、为便于调试,每个函数和过程应该尽可能短小精悍,并配合具体详细的注释。
4>、应多利用try-finally来处理可能出现的错误和异常,注意这时要引用SysUtils单元。
5>、尽可能少引用单元以减小DLL的大小,特别是不要引用可视化单元,如Dialogs单元。例如一般情况下,我们可以不
引用Classes单元,这样可使编译后的DLL减小大约16Kb。
2、调用技巧:
1>、在用静态方法时,可以给被调用的函数或过程更名。改写引用函数为
function TestC(i:integer):integer;stdcall; external 'Project1.dll ' name 'TestDll ';
其中name的作用就是重命名(原名称仍然大小写敏感)。
直接通过名称调用(注意名称大小写敏感)。
function TestDll (i:integer):integer;stdcall; external 'Project1.dll ' ;
// 如果定义了Index就可以使用,通过索引号调用。程序中可以用与DLL中不一样的名称.
procedure test2;external 'Project1.dll' index 1; // exports TestDll index 1;
2>、可把我们编写的DLL放到Windows目录下或者Windows\system目录下。这样做可以在external语句中或LoadLibrary
语句中不写路径而只写DLL的名称。但这样做有些不妥,这两个目录下有大量重要的系统DLL,如果您编的DLL与
它们重名的话其后果简直不堪设想.
3、调试技巧:
1>、我们知道DLL在编写时是不能运行和单步调试的。有一个办法可以,那就是在Run|parameters菜单中设置一个宿
主程序。在Local页的Host Application栏中添上宿主程序的名字。宿主程序是使用它生成的DLL包的程序。然后
再DLL工程中点击【Run】就可进行单步调试、断点观察和运行了。
2>、添加DLL的版本信息。如果包含了版本信息,DLL的大小会增加2Kb。增加这么一点空间是值得的。很不幸我们如
果直接使用Project|options菜单中Version选项是不行的,还必须增加{$R *.res},才会显示版本信息;
3>、为了避免与别的DLL重名,在给自己编写的DLL起名字的时候最好采用字符数字和下划线混合的方式。如:jl_try16.dll。
八、具体的一个例子:用DLL文件封装窗体的实现方法实例:
一个程序不再是单一的一个EXE文件了,而是由一个EXE文件加N个DLL文件组成,这样做的原因是方
便以后的维护与更新,也是跨平台开发的重要一步。
1、打开DELPHI,新建一个Dll Wizard
2、 在新建的Dll里新建一个Form
3、 在新建的Form里uses stdctrls
4、 在var下面写:
Procedure synapp(App:THandle);stdcall;
Procedure showform;stdcall;
5、然后在implementation 下面uses math
6、 在{$R *.dfm}下面写
Procedure synapp(App:THandle);stdcall;
Begin
Application.Handle:=app;// 防止每显示一个窗体,就在任务栏中显示一个图标
End;
Procedure showform;stdcall;
Begin
Form1:=Tform1.create(application);
Form1.show;
End;
7 、在dll的Library文件里的{$R *.res}下面写:
exports
Sysapp,show;
上面到此为止完成了DLL封装窗体的创建
8、下面是调用了
1> 、 在要调用DLL文件的程序的var下写:
Procedure synapp(App:THandle);stdcall;external ‘my.dll’ ;//----你的DLL文件名
Procedure showform;stdcall;external‘my.dll’;//----你的DLL文件名
注:把你写好的DLL放在本程序的同一目录下,和上面一样,要uses math;
2> 、在你的程序的Button的On Click事件下写:
Synapp(applicatiln.Handle);
Showform;
完毕
用DLL文件封装窗体,每一个DLL工程中的窗体都是独立的一个进程。所以任何操作都是独立的。在DLL
工程中使用RegisterClass方法对窗体进行祖册是,在应用程序工程或者其他工程再用FindClass方法查找这个类是无
效的。而对于DLL工程而言,方法指针的传递非常的安全,所以可以维护一个指针列表,用于指向各个DLL工程中
FindClass方法的地址。在需要查找窗体类时,对所以的DLL工程的FindClass方法进行调用即可。
封装在DLL工程中的窗体,每打开一次窗体就会出现一个图标在任务栏区。为了解决这个问题,应在调用
DLL文件时,将应用程序中的Application对象和Screen对象传到DLL工程中,并替换DLL工程中这两个对象。
转自:http://blog.csdn.net/zang141588761/article/details/51248258