第十章-动态链接库编程(一)(3)

10.2.2 调用DLLs

有两种方法可用于调用一个储存在DLLs中的过程。

1.静态调用或显示装载

使用一个外部声明子句,使DLLs在应用程序开始执行前即被装入。例如: 

function Instr(SourceStr : PChar;Check : Char); Integer; far; external 'UseStr';

使用这种方法,程序无法在运行时间里决定DLLs的调用。假如一个特定的DLLs在运行时无法使用,则应用程序将无法执行。

2.动态调用或隐式装载

使用Windows API函数LoadLibray和GetProcAddress可以实现在运行时间里动态装载DLLs并调用其中的过程。

若程序只在其中的一部分调用DLLs的过程,或者程序使用哪个DLLs, 调用其中的哪个过程需要根据程序运行的实际状态来判断,那么使用动态调用就是一个很好的选择。

使用动态调用,即使装载一个DLLs失败了,程序仍能继续运行。 

10.2.3 静态调用

在静态调用一个DLLs中的过程或函数时,external指示增加到过程或函数的声明语句中。被调用的过程或函数必须采用远调用模式。这可以使用far过程指示或一个{$F +}编译指示。

Delphi全部支持传统Windows动态链接库编程中的三种调用方式,它们是:

● 通过过程/函数名

● 通过过程/函数的别名

● 通过过程/函数的顺序号 

通过过程或函数的别名调用,给用户编程提供了灵活性,而通过顺序号(Index)调用可以提高相应DLL的装载速度。 

10.2.4 动态调用 

10.2.4.1 动态调用中的API函数 

动态调用中使用的Windows API函数主要有三个,即:Loadlibrary,GetProcAddress和Freelibrary。

1.Loadlibrary: 把指定库模块装入内存

语法为: 

function Loadlibrary(LibFileName: PChar): THandle; 

LibFileName指定了要装载DLLs的文件名,如果LibFileName没有包含一个路径,则Windows按下述顺序进行查找:

(1)当前目录;

(2)Windows目录(包含win.com的目录)。函数GetWindowDirectory返回这一目录的路径;

(3)Windows系统目录(包含系统文件如gdi.exe的目录)。函数GetSystemDirectory返回这一目录的路径;

(4)包含当前任务可执行文件的目录。利用函数GetModuleFileName可以返回这一目录的路径;

(5)列在PATH环境变量中的目录;

(6)网络的映象目录列表。

如果函数执行成功,则返回装载库模块的实例句柄。否则,返回一个小于HINSTANCE_ERROR的错误代码。错误代码的意义如下表: 

表10.2 Loadlibrary返回错误代码的意义

━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━

错误代码         意        义

——————————————————————————————————————

  0 系统内存不够,可执行文件被破坏或调用非法

  2 文件没有被发现

  3 路径没有被发现

  5 企图动态链接一个任务或者有一个共享或网络保护错

  6 库需要为每个任务建立分离的数据段

  8 没有足够的内存启动应用程序

10 Windows版本不正确

  11 可执行文件非法。或者不是Windows应用程序,或者在.EXE映

   像中有错误

  12 应用程序为一个不同的操作系统设计(如OS/2程序)

13 应用程序为MS DOS4.0设计

  14 可执行文件的类型不知道

  15 试图装载一个实模式应用程序(为早期Windows版本设计)

16 试图装载包含可写的多个数据段的可执行文件的第二个实例

  19 试图装载一个压缩的可执行文件。文件必须被解压后才能被装裁

  20 动态链接库文件非法

  21 应用程序需要32位扩展

━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━

假如在应用程序用Loadlibrary调用某一模块前,其它应用程序已把该模块装入内存,则Loadlibrary并不会装载该模块的另一实例,而是使该模块的“引用计数”加1。 

2.GetProcAddress:捡取给定模块中函数的地址

语法为: 

function GetProcAddress(Module: THandle; ProcName: PChar): TFarProc; 

Module包含被调用的函数库模块的句柄,这个值由Loadlibrary返回。如果把Module设置为nil,则表示要引用当前模块。

ProcName是指向含有函数名的以nil结尾的字符串的指针,或者也可以是函数的次序值。如果ProcName参数是次序值,则如果该次序值的函数在模块中并不存在时,GetProcAddress仍返回一个非nil的值。这将引起混乱。因此大部分情况下用函数名是一种更好的选择。如果用函数名,则函数名的拼写必须与动态链接库文件EXPORTS节中的对应拼写相一致。

如果GetProcAddress执行成功,则返回模块中函数入口处的地址,否则返回nil。

3.Freelibrary:从内存中移出库模块

语法为: 

procedure Freelibrary(Module : THandle); 

Module为库模块的句柄。这个值由Loadlibrary返回。

由于库模块在内存中只装载一次,因而调用Freelibrary首先使库模块的引用计数减一。如果引用计数减为0,则卸出该模块。

每调用一次Loadlibrary就应调用一次FreeLibray,以保证不会有多余的库模块在应用程序结束后仍留在内存中。 

10.2.4.2 动态调用举例 

对于动态调用,我们举了如下的一个简单例子。系统一共包含两个编辑框。在第一个编辑框中输入一个字符串,而后在第二个编辑框中输入字符。如果该字符包含在第一个编辑框的字符串中,则标签框显示信息:“位于第n位。”,否则显示信息:“不包含这个字符。”。如图是程序的运行界面。

输入检查功能的实现在Edit2的OnKeyPress事件处理过程中,程序清单如下。 

procedure TForm1.Edit2KeyPress(Sender: TObject; var Key: Char);

var

order: Integer;

txt: PChar;

PFunc: TFarProc;

Moudle: THandle;

begin

Moudle := Loadlibrary('c:\dlls\example.dll');

if Moudle > 32 then

begin

Edit2.text := '';

Pfunc := GetProcAddress(Moudle,'Instr');

txt := StrAlloc(80);

txt := StrPCopy(txt,Edit1.text);

Order := TInstr(PFunc)(txt,Key);

if Order = -1 then

Label1.Caption := '不包含这个字符 '

else

Label1.Caption := '位于第'+IntToStr(Order+1)+'位';

end;

Freelibrary(Moudle);

end;

在利用GetProcAddess返回的函数指针时,必须进行强制类型转换: 

Order := TInstr(PFunc)(text,Key);

TInStr是一个定义好了的函数类型: 

type

TInStr = function(Source: PChar;Check: Char): Integer; 

时间: 2024-09-24 12:48:29

第十章-动态链接库编程(一)(3)的相关文章

第十章-动态链接库编程(二)(4)

10.4.2.2 Delphi应用程序调用重用窗体 在Delphi应用程序中调用重用窗体,首先必须包含passform.dll的两个输出函数: function GetPassword(Password: PChar): Boolean; far; external 'c:\dlls\PassForm'; function SetPassword(PassWord: PChar): Boolean; far; external 'c:\dlls\PassForm'; 这位于程序单元的implem

第十章-动态链接库编程(一)(2)

10.2.1.3 DLLs中的变量和段 一个DLLs拥有自己的数据段(DS),因而它声明的任何变量都为自己所私有.调用它的模块不能直接使用它定义的变量.要使用必须通过过程或函数界面才能完成.而对DLLs来说,它永远都没有机会使用调用它的模块中声明的变量. 一个DLLs没有自己的堆栈段(SS),它使用调用它的应用程序的堆栈.因此在DLL中的过程.函数绝对不要假定DS = SS.一些语言在小模式编译下有这种假设,但使用Delphi可以避免这种情况.Delphi绝不会产生假定DS = SS的代码,De

第十章-动态链接库编程(一)(1)

动态链接库(DLLs)是从C语言函数库和Pascal库单元的概念发展而来的.所有的C语言标准库函数都存放在某一函数库中,同时用户也可以用LIB程序创建自己的函数库.在链接应用程序的过程中,链接器从库文件中拷贝程序调用的函数代码,并把这些函数代码添加到可执行文件中.这种方法同只把函数储存在已编译的.OBJ文件中相比更有利于代码的重用. 但随着Windows这样的多任务环境的出现,函数库的方法显得过于累赘.如果为了完成屏幕输出.消息处理.内存管理.对话框等操作,每个程序都不得不拥有自己的函数,那么W

第十章-动态链接库编程(二)(1)

10.3.2.2 服务器程序的编写 服务器程序必须包含对DLL的调用代码,如: function GetGlobalMem: THandle; far; external 'c:\dlls\glbmem'; 通过调用该函数,服务器可以获得全局内存块的句柄. 在写入数据前,服务器必须锁定全局内存,以避免在写入过程中Windows移动该内存块的位置. 函数GlobalLock锁定全局内存并返回指向该内存块的指针: pMem := GlobalLock(hMem); 对pMem的任何修改都会反映到全局

第十章-动态链接库编程(二)(3)

在口令设置窗口中,为了确保用户记住了设置的口令,在用户输入并按回车键后,要求用户再次输入进行确认.只有用户重新输入的字符串与原设置口令相同,口令设置窗口才能正常关闭 .否则将原设置口令清空,要求用户再次输入.以上功能的实现在编辑框的OnKeyPress事件处理过程中. procedure TSetPassWordForm.Edit1KeyPress(Sender: TObject; var Key: Char); begin if Edit1.text = '' then Exit; if Ke

第十章-动态链接库编程(二)(2)

10.4.1 利用DLLs实现窗体重用的一般步骤 利用DLLs实现窗体重用的步骤是: 1.在集成开发环境(IDE)中,按自己的需要设计一个窗体: 2.编写一个用于输出的函数或过程.在该函数或过程中,设计的窗体被实例化: 3.重复步骤1.2,直到完成所有重用窗体的设计: 4.打开工程文件,进行修改,以适应生成 .dll文件的需要: (1).把保留字program设为library: (2).从uses子句中去掉Forms单元: (3).移去begin,end之间的所有代码: (4).在uses子句

第十章-动态链接库编程(一)(4)

10.3 利用DLLs实现数据传输 10.3.1 DLLs中的全局内存 Windows规定:DLLs并不拥有它打开的任何文件或它分配的任何全局内存块.这些对象由直接或间接调用DLLs的应用程序拥有.这样,当应用程序中止时,它拥有的打开的文件自动关闭,它拥有的全局内存块自动释放.这就意味着保存在DLLs全局变量中的文件和全局内存块变量在DLLs没有被通知的情况下就变为非法.这将给其它使用该DLLs的应用程序造成困难. 为了避免出现这种情况,文件和全局内存块句柄不应作为DLLs的全局变量,而是作为D

delphi入门教程

第一章-Delphi入门(一)(1) 第一章-Delphi入门(一)(2) 第一章-Delphi入门(一)(3) 第一章-Delphi入门(二)(1) 第一章-Delphi入门(二)(2) 第一章-Delphi入门(二)(3) 第一章-Delphi入门(二)(4) 第一章-Delphi入门(三)(1) 第一章-Delphi入门(三)(2) 第一章-Delphi入门(三)(3) 第一章-Delphi入门(三)(4) 第一章-Delphi入门(三)(5) 第一章-Delphi入门(三)(6) 第一章

112_《Delphi2高级程序设计指南》

<Delphi2高级程序设计指南> Delphi 教程 系列书籍 (112) <Delphi2高级程序设计指南> 网友(邦)整理 EMail: shuaihj@163.com 下载地址: Part1 Part2 Part3 Part4 Part5 Part6 作者: 姚庭宝 出版社: 电子工业出版社 内容简介 Delphi 2.0高级程序设计指南(非常的优秀) 目录 第一部分 基础篇 第一章 Delphi快速入门 第二章 Delphi面向对象的编程方法 第三章 字符串列表及应用 第