delphi 基础之三 编写和调用dll文件

时间:2023-03-08 19:25:09
delphi 基础之三  编写和调用dll文件

delphi 编写和调用dll文件

 

Windows 的执行文件可以划分为两种形式程序和动态连接库 (DLLs)。一般程序运行是用.EXE文件,但应用程序有时也可以调用存储在DLL的函数。

在如下几种情况下,调用DLL 是合理的:

1) 不同的程序使用相同的DLL ,这样只需要将DLL 在内存中装载一次,节省了内存的开销。

2) 当某些内容需要升级的时候,如果使用DLL 只需要改变DLL 就可以了,而不需要把整个程序都进行变动。

3) 由于DLL 是独立于语言的,所以,当不同语言习惯的人共同开发一个大型项目的时候,使用DLL 便于程序系统的交流,当然,Delphi开发的DLL 也可以在诸如Visual BASIC,C++ 等系统中使用。

下面通过几个例子,说明Delphi开发动态连接库的方法和规范。

第一节 动态连接库的构建和调用方法 

一、动态连接库构建

1.创建了一个动态连接库的基本模块

File---New---Other---DLL Wizard

library Project1;

{如果DLL输出的过程或函数带有长字符类型的参数,或者函数返回类型是长字符串或带有长字符串元素的构造类型,Object Pascal规定无论是DLL还是调用它的程序必须把ShareMem单元加到Uses部分。而ShareMem单元是从DelphiMM.DLL这个DLL中引入的接口单元,因此这种程序分发时必须带有DelphiMM.DLL。Delphi建议为了避免使用DELPHIMM.DLL,传递字符串信息时使用PChar或ShortString类型参数。(原文为英文) }

uses 

SysUtils,  Classes;

 

代码部分)

 

 {$R *.res} 

begin 

end. 

把工程名改为Math(Save Project  as …),并写入必要的函数

library Math;

uses

  SysUtils,

  Classes;

 

function Mround (d:Double ):Double ;stdcall;   //四舍五入函数:数学上的四舍五入函数不begin                                       //同于Delphi的银行家四舍五入法

Result:=Int(d*100+0.5)/100;

end;

exports

Mround ;

{$R *.RES}

begin

end.

从这个例子中可以看出DLL 程序的几个规则:

1) 在DLL 程序中,输出函数必须被声明为stdcall,以使用标准的Win32 参数传递技术来代替优化的Register。参数的调用约定。调用约定如下:

指令  传递顺序  参数删除

stdcall 从左到右 函数方面

cdecl  从右到左 调用方面

Pascal  从左到右 函数方面

register 从左到右 函数方面

(说明:

register,默认,即是 唯一使用 CPU寄存器的参数传递方式,也是传递速度最快的方式; pascal: 调用协议仅用于向后兼容,即向旧的版本兼容;
cdecl: 多用于 C和 C++语言编写的例程,也用于需要由调用者清除参数的例程; stdcall: 和safecall主要用于调用Windows API 函数;其中safecall还用于双重接口。

2)所有的输出函数都必须列在exports子句下面,这使的子例程在DLL外部就可以看到

exports

Mround[name ‘别名’] ;

列出了用户使用这个函数的接口名字。虽然别名不是必须的,但最好给个别名,以便用户程序更容易找到这个函数,同时还要指出Delphi 6.0取消了Delphi 5.0中允许使用的index 如果还用Index指明接口名字,Delphi 6.0中将提示错误。

实例中给出了messagebox提示方法,主要想说明一个问题:

而messagebox(0,’’,’’,mb_ok) 是Windows提供的API 函数,做出的程序会比较小。

Showmessage(‘’),是VCL 提供的函数,由于多次编译VCL,做出的程序会比较大。

这就是说,编写DLL 程序的时候,要尽量避免多次编译VCL 。作为一个实例,这里把两种方法都列出来了。

2.保存

3.编译:Projrct---Build Math.DLLl

这就完成了一个简单的动态连接库的编写。

二、             动态连接库的调用

1.静态调用

首先 (如果在窗体中要求在implementation下声明,本例直接在程序文件dpr中运行,不带窗体)做调用声明 function Mround(d: Double): Double; stdcall; external 'Math.dll'; 

完整调用程序如下:

program test;

uses

Windows,

Messages,

SysUtils,

Forms,

Dialogs,

StdCtrls;

var

d: Double;

info: Double;

{$R *.res}

begin

Application.Initialize;                             /程序初始化并开始运行

d := StrToFloat(inputbox('输入窗口', '请输入要四舍五入的数字', '0'));

MessageBox(0, pchar(floattostr(Mround(d))), '结果', MB_OK orMB_ICONINFORMATIO

{调用dll函数}

Application.Run;

end.

 

2.动态调用(实例)

Type                                                                          TShowForm = function (AHandle: THandle; ACaption: String): BOOL; StdCall;               {在此创建一个函数类}                                                                         EDLLLoadError = class(Exception);//同时创建一个出错记录类
                                                                                             TMAINCLTR = class(TForm) //这里不变,系统自动生成(调用Dll窗体的应用程序的窗体类)

private

{ Private declarations }

public

{ Public declarations }

end;

var

MAINCLTR1: TMAINCLTR;

implementation

{$R *.DFM}
//函数、过程定义
Procedure TMAINCLTR.ToolButton1Click(Sender: TObject); //按钮的调用事件:调用过程
var LibHandle: THandle;
ShowForm: TShowForm;
begin
Application.Title:=' DLL文件测试程式';
{尝试装入DLL文件}
LibHandle := LoadLibrary('MGRFORM.DLL');
try
if LibHandle = 0 then
raise EDLLLoadError.Create('Unable to Load DLL(无法成功装入MGRFORM.DLL)');
@ShowForm := GetProcAddress(LibHandle, 'ShowForm');
if not (@ShowForm = nil) then
ShowForm(Application.Handle, '人事资料管理')//呼叫出窗体
else
RaiseLastWin32Error;
finally
FreeLibrary(LibHandle); // Unload the DLL.
end;
end;
============== END ==================

第二节 DLL 中的Delphi窗体 

一、利用DLLs实现窗体重用的步骤是:

1.在集成开发环境(IDE)中,按自己的需要设计一个窗体;

2.编写一个用于输出的函数或过程。在该函数或过程中,设计的窗体被实例化

方法是:

1)首先按普通方法制作窗体,不过在interface区域,对接口函数做如下声明

function Createform(capt:string):string;stdcall;

2)在implementation下加入接口函数

function Createform(capt:string):string;stdcall; 

var Form1: TForm1; 

begin 

form1:=Tform1.Create(application);               设计的窗体被实例化

传递一个Application参数,用它建立Form.}            

 form1.show;                                  

form1.caption:=capt; 

end; 

3.重复步骤1、2,直到完成所有重用窗体的设计;

4.打开File---New---Other---DLL Wizard建立Dll文件。添加保留字exprots。exports 后是输出函数名或过程名。(生成窗体)

5.编译生成DLLs文件;要声明:

uses 

unit1 in ’unit1.pas’; 

exports                  {写入接口标示符} 

Createform name ’Myform’;

6.在其它应用程序中调用重用窗体。

重用窗体的调用同一般DLLs函数或过程的调用完全一致

 

 

DLL文件在Delphi的创建及调用

一.函数过程的写法:

library ForstDLL;

uses
SysUtils,
Classes;

{$R *.RES}
// 1.定义函数具体过程和输出接口方式
// --------------------------------
// 函数 1
// 功能:数据3倍放大函数
// --------------------------------
function PenniesToSoins(SourceResult:Integer):Integer;stdCall;
begin
if SourceResult>0 then
Result:=SourceResult*3 //结果存放于Result
else
Result:=SourceResult;
end;

exports
PenniesToSoins; //2.函数输出口定义

end.

二.在DLL中创建Form
=======================
1.创建DLL工程,及加入设置好的Form

library MGRFORM;
uses
SysUtils,
Classes,
MGRPERFM in 'MGRPERFM.pas' {FormPERSON};//1.Form的代码(与一般的Form一样)

{$R *.RES}
exports
ShowForm;//2.函数输出口定义
begin
end.

2. 在DLL设定的Form的设置
===========================================
unit MGRPERFM;

interface

uses
Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs,
ComCtrls, ToolWin, ImgList;

type
TFormPERSON = class(TForm)
private
{ Private declarations }
public
{ Public declarations }
end;

//此变量不再用,给其改个地方,放在implementation中。如下(改变之一)
//var
// FormPERSON: TFormPERSON; {宣布Form函数出口}//改变之二

function ShowForm(AHandle: THandle; ACaption: String):BOOL; StdCall;

{窗体实例化函数}

implementation

{$R *.DFM}
//函数、过程定义
function ShowForm(AHandle: THandle; ACaption: String):BOOL;
var
FormPERSON: TFormPERSON; //定义窗体类(上面的放到了此处)
begin
// Copy application handle to DLL's TApplication object
//拷贝应用程序句柄给DLL的应有程序对象
Application.Handle := AHandle;
FormPERSON := TFormPERSON.Create(Application);//创建控件TForm
try
FormPERSON.Caption := ACaption;
FormPERSON.ShowModal;//显示此Form
// Pass the date back in Result
Result := False; //返回成功值
finally
FormPERSON.Free;
end;
end;

三.DLL中函数及窗体的调用
==========================
1.调用方法一:静态引用

implementation //在此的下方写明调用函数的DLL

{$R *.DFM}
//引用DLL内函数
function PenniesToSoins(SourceResult:Integer):Integer;
StdCall;external 'FORSTDLL.DLL';

........

2.调用方法二:动态引用
==============
type //在此创建一个函数类
// -------------------------------
{ Forst, define a procedural data type, this should reflect the
procedure that is exported from the DLL. }
{ Create a new exception class to reflect a failed DLL load }

TShowForm = function (AHandle: THandle; ACaption: String): BOOL; StdCall;

EDLLLoadError = class(Exception);//同时创建一个出错记录类
//  -------------------------------
TMAINCLTR = class(TForm) //这里不变,系统自动生成

......

procedure TMAINCLTR.ToolButton1Click(Sender: TObject);
var //按钮的调用事件:调用过程
LibHandle: THandle;
ShowForm: TShowForm;
begin
Application.Title:=' DLL文件测试程式';
{尝试装入DLL文件}
LibHandle := LoadLibrary('MGRFORM.DLL');
try
if LibHandle = 0 then
raise EDLLLoadError.Create('Unable to Load DLL(无法成功装入MGRFORM.DLL)');
@ShowForm := GetProcAddress(LibHandle, 'ShowForm');
if not (@ShowForm = nil) then
ShowForm(Application.Handle, '人事资料管理')//呼叫出窗体
else
RaiseLastWin32Error;
finally
FreeLibrary(LibHandle); // Unload the DLL.
end;
end;
============== END ==================