静态调用或显示装载
使用一个外部声明子句使DLLs在应用程序开始执行前即被装入例如
function Instr(SourceStr : PChar;Check : Char) Integer; far; external UseStr;
使用这种方法程序无法在运行时间里决定DLLs的调用假如一个特定的DLLs在运行时无法使用则应用程序将无法执行
动态调用或隐式装载
使用Windows API函数LoadLibray和GetProcAddress可以实现在运行时间里动态装载DLLs并调用其中的过程
若程序只在其中的一部分调用DLLs的过程或者程序使用哪个DLLs 调用其中的哪个过程需要根据程序运行的实际状态来判断那么使用动态调用就是一个很好的选择
使用动态调用即使装载一个DLLs失败了程序仍能继续运行
静态调用
在静态调用一个DLLs中的过程或函数时external指示增加到过程或函数的声明语句中被调用的过程或函数必须采用远调用模式这可以使用far过程指示或一个{$F +}编译指示
Delphi全部支持传统Windows动态链接库编程中的三种调用方式它们是
● 通过过程/函数名
● 通过过程/函数的别名
● 通过过程/函数的顺序号
通过过程或函数的别名调用给用户编程提供了灵活性而通过顺序号(Index)调用可以提高相应DLL的装载速度
动态调用
动态调用中的API函数
动态调用中使用的Windows API函数主要有三个即LoadlibraryGetProcAddress和Freelibrary
Loadlibrary: 把指定库模块装入内存
语法为
function Loadlibrary(LibFileName: PChar) THandle;
LibFileName指定了要装载DLLs的文件名如果LibFileName没有包含一个路径则Windows按下述顺序进行查找
()当前目录
()Windows目录(包含wincom的目录)函数GetWindowDirectory返回这一目录的路径
()Windows系统目录(包含系统文件如gdiexe的目录)函数GetSystemDirectory返回这一目录的路径
()包含当前任务可执行文件的目录利用函数GetModuleFileName可以返回这一目录的路径
()列在PATH环境变量中的目录
()网络的映象目录列表
如果函数执行成功则返回装载库模块的实例句柄否则返回一个小于HINSTANCE_ERROR的错误代码错误代码的意义如下表
表 Loadlibrary返回错误代码的意义
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
错误代码 意 义
系统内存不够可执行文件被破坏或调用非法
文件没有被发现
路径没有被发现
企图动态链接一个任务或者有一个共享或网络保护错
库需要为每个任务建立分离的数据段
没有足够的内存启动应用程序
Windows版本不正确
可执行文件非法或者不是Windows应用程序或者在EXE映像中有错误
应用程序为一个不同的操作系统设计(如OS/程序)
应用程序为MS DOS设计
可执行文件的类型不知道
试图装载一个实模式应用程序(为早期Windows版本设计)
试图装载包含可写的多个数据段的可执行文件的第二个实例
试图装载一个压缩的可执行文件文件必须被解压后才能被装裁
动态链接库文件非法
应用程序需要位扩展
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
假如在应用程序用Loadlibrary调用某一模块前其它应用程序已把该模块装入内存则Loadlibrary并不会装载该模块的另一实例而是使该模块的引用计数加
GetProcAddress:捡取给定模块中函数的地址
语法为
function GetProcAddress(Module: THandle; ProcName: PChar) TFarProc;
Module包含被调用的函数库模块的句柄这个值由Loadlibrary返回如果把Module设置为nil则表示要引用当前模块
ProcName是指向含有函数名的以nil结尾的字符串的指针或者也可以是函数的次序值如果ProcName参数是次序值则如果该次序值的函数在模块中并不存在时GetProcAddress仍返回一个非nil的值这将引起混乱因此大部分情况下用函数名是一种更好的选择如果用函数名则函数名的拼写必须与动态链接库文件EXPORTS节中的对应拼写相一致
如果GetProcAddress执行成功则返回模块中函数入口处的地址否则返回nil
Freelibrary:从内存中移出库模块
语法为
procedure Freelibrary(Module : THandle)
Module为库模块的句柄这个值由Loadlibrary返回
由于库模块在内存中只装载一次因而调用Freelibrary首先使库模块的引用计数减一如果引用计数减为则卸出该模块
每调用一次Loadlibrary就应调用一次FreeLibray以保证不会有多余的库模块在应用程序结束后仍留在内存中
动态调用举例
对于动态调用我们举了如下的一个简单例子系统一共包含两个编辑框在第一个编辑框中输入一个字符串而后在第二个编辑框中输入字符如果该字符包含在第一个编辑框的字符串中则标签框显示信息位于第n位否则显示信息不包含这个字符如图是程序的运行界面
输入检查功能的实现在Edit的OnKeyPress事件处理过程中程序清单如下
procedure TFormEditKeyPress(Sender: TObject; var Key: Char)
var
order: Integer;
txt: PChar;
PFunc: TFarProc;
Moudle: THandle;
begin
Moudle := Loadlibrary(c:\dlls\exampledll)
if Moudle > then
begin
Edittext := ;
Pfunc := GetProcAddress(MoudleInstr)
txt := StrAlloc()
txt := StrPCopy(txtEdittext)
Order := TInstr(PFunc)(txtKey)
if Order = then
LabelCaption := 不包含这个字符
else
LabelCaption := 位于第+IntToStr(Order+)+位;
end;
Freelibrary(Moudle)
end;
在利用GetProcAddess返回的函数指针时必须进行强制类型转换
[] [] [] []