关闭

Win32下动态链接库(DLL)编程原理的深探

284人阅读 评论(0) 收藏 举报

主题:DLL是Windows最重要的组成要素,Windows中的许多新功能、新特性都是通过DLL来
实现的,因此掌握它、应用它是非常重要的。
动态链接库不仅可以作为一个运行模块,包括函数代码,而且可以包含程序以外的任何数据
或资源(位图、图标等等)。动态链接库就是给应用程序提供函数或者资源。
DLL是一种磁盘文件(通常带有DLL扩展名),它由全局数据、服务函数和资源组成,
在运行时被系统加载到进程的虚拟空间中,成为调用进程的一部分。在运行时,只有当EXE
程序确实要调用这些DLL模块的情况下,系统才会将它们装载到内存空间中。每个进程都复
制了自己的读/写全局变量。如果想要与其它进程共享内存,必须使用内存映射文件或者声
明一个共享数据段。DLL模块需要的堆栈内存都是从运行进程的堆栈中分配出来的。
DLL文件中包含一个导出函数表。这些导出函数由它们的符号名和称为标识号的整数与外界联系起来。


导入导出函数:
在DLL代码中,必须像下面这样明确声明导出函数:
__declspec(dllexport) int MyFunction(int n);
但也可以在模块定义(DEF)文件中列出导出函数,不过这样做常常引起更多的麻烦。在应用程序方面,要
求像下面这样明确声明相应的输入函数:
__declspec(dllimport) int MyFuncition(int n); (这是隐式链接dll)
仅有导入和导出声明并不能使应用程序内部的函数调用链接到相应的DLL文件上。应用程序的项目必须为
链接程序指定所需的输入库(LIB文件)。而且应用程序事实上必须至少包含一个对DLL函数的调用。
隐式链接和显式链接dll

显式:typedef double(SQRTPROC)(double);

HINSTANCE hInstance; //设置全局变量用于存储DLL句柄
SQRTPROC* pFunction; 第二个变量ShowMe是指向DLL,库中函数的指针
VERIFY(hInstance=::LoadLibrary("c://winnt//system32//mydll.dll"));
VERIFY(pFunction=(SQRTPROC*)::GetProcAddress(hInstance,"SquareRoot"));
double d=(*pFunction)(81.0);//调用该DLL函数
隐式: 程序员通过静态链接方式编译生成应用程序时,应用程序中的调用函数与LIB文件中导出符号相
匹配,这些符号或标识号进入到生成的EXE文件中.
隐式链接和显式链接dll的区别:
在隐式链接方式中,所有被应用程序调用的DLL文件都会在应用程序EXE文件加载时被加载在到内存中;但
如果采用显式链接方式,程序员可以决定DLL文件何时加载或不加载。

DLLMAIN函数:
当Windows加载DLL模块时调用这一函数。系统首先调用全局对象的构造函数,然后调用全局函数DLLMain。
DLLMain函数不仅在将DLL链接加载到进程时被调用,在DLL模块与进程分离时(以及其它时候)也被调用。
进程中的每个DLL模块被全局唯一的32字节的HINSTANCE句柄标识。进程自己还有一个HINSTANCE句柄。所
有这些模块句柄都只有在特定的进程内部有效,它们代表了DLL或EXE模块在进程虚拟空间中的起始地址。
寻找DLL的路径:
Windows将遵循下面的搜索顺序来定位DLL:
1. 包含EXE文件的目录,
2. 进程的当前工作目录,
3. Windows系统目录,
4. Windows目录,
5. 列在Path环境变量中的一系列目录。


要点:

一:DLL文件的省缺名称是与工程名一致的(也是在.DEF文件中LIBRARY 之后的名字),不要试图在制作完毕之后通
过简单的修改.DLL文件的文件名来改变它,这会导致使用该DLL的应用程序错误.

DLL中定义有两种函数:
导出函数(exportfunction): 可以被其他模块调用
内部函数(internalfunction): 只能在DLL内部使用

基于MFC的DLL不适用与制作读取二进制文件的DLL (不易移植,无法正确读取与DOS应用程序共享的二进制文件)
基于API的DLL可以正确读取在DOS环境中创建的二进制文件.
纯标准函数dll可以在很多平台使用!(移植性最强)
为了让其它语言使用:除非你绝对需要使用C++编码,否则我还是推荐使用C来写DLL。
VC下的DLL类型:
Non-MFC DLL:指的是不用MFC的类库结构,直接用C语言写的DLL,其输出的函数一
般用的是标准C接口,并能被非MFC或MFC编写的应用程序所调用。LL,

Regular DLL:和下述的Extension Dlls一样,是用MFC类库编写的。明显的特点是
在源文件里有一个继承CWinApp的类。其又可细分成静态连接到MFC和动态连接到MFC上
的。但静态连接到MFC的动态连接库只被VC的专业般和企业版所支持。

Extension DLL:用来实现从MFC所继承下来的类的重新利用,也就是说,用这种类
型的动态连接库,可以用来输出一个从MFC所继承下来的类。Extension DLL使用MFC的
动态连接版本所创建的,并且它只被用MFC类库所编写的应用程序所调用。
用来实现从MFC所继承下来的类的重新利用,也就是说,用这种类型的动态连接库,可以用
来输出一个从MFC所继承下来的类。它输出的函数仅可以被使用MFC且动态链接到MFC的应用
程序使用。可以从MFC继承你所想要的、更适于你自己用的类,并把它提供给你的应用程序。
你也可随意的给你的应用程序提供MFC或MFC继承类的对象指针。Extension DLL使用MFC的动
态连接版本所创建的,并且它只被用MFC类库所编写的应用程序所调用。Extension DLLs 和
Regular DLLs不一样,它没有一个从CWinApp继承而来的类的对象,所以,你必须为自己DllMain
函数添加初始化代码和结束代码。
和规则DLL相比,有以下不同:

1、它没有一个从CWinApp派生的对象;
2、它必须有一个DllMain函数;
3、DllMain调用AfxInitExtensionModule函数,必须检查该函数的返回值,如果返回0,DllMmain也返回0;
4、如果它希望输出CRuntimeClass类型的对象或者资源(Resources),则需要提供一个初始化函数来创建一个CDynLinkLibrary对象。并且,有必要把初始化函数输出;
5、使用扩展DLL的MFC应用程序必须有一个从CWinApp派生的类,而且,一般在InitInstance里调用扩展DLL的初始化函数。

所有从DLL输出的函数应该以如下语句开始:
AFX_MANAGE_STATE(AfxGetStaticModuleState( ))
此语句用来正确地切换MFC模块状态。

在制作DLL时,VC++对函数的省缺声明是"__cedcl",也就是说,如果你在声明你的函数时不作特殊声明的话,你
制作的DLL将只能被C/C++调用,如果你想用其他开发语言(比如VB5.0)调用它就会报错,即使调用方法完全正确.

二:声明类型详解:

三:制DLL所要的文件:
模 块 定 义 文 件(.DEF) 是 由 一 个 或 者 多 个 用 于 描 述DLL 属 性 的 模 块 语 句 组 成
的 文 本 文 件, 每 个.DEF 文 件 至 少 必 须 包 含 以 下 模 块 定 义 语 句:
第 一 个 语 句 必 须 是LIBRARY 语 句, 指 出DLL 的 名 字。
EXPORTS 语 句 列 出 被 导 出 函 数 的 名 字。
可 以 使 用DESCRIPTION 语 句 描 述DLL 的 用 途( 此 句 可 选)。
";" 对 一 行 进 行 注 释( 可 选)
实 现 文 件(.cpp 文 件 为 例)
实 现 入 口 表 函 数 的.cpp 文 件 中, 包 含DLL 入 口 点 处 理 的API 函 数 和 导 出 函 数 的 代 码。

四:
介绍Non-MFC DLLs的编写方法:如果用VC的向导那就很方便了!
写法:
BOOL APIENTRY DllMain(HANDLE hModule,DWORD ul_reason_for_call,

LPVOID lpReserved)

{

switch( ul_reason_for_call ) {

case DLL_PROCESS_ATTACH:

.......

case DLL_THREAD_ATTACH:

.......

case DLL_THREAD_DETACH:

.......

case DLL_PROCESS_DETACH:

.......

}

return TRUE;

}
 

0
0

查看评论
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
    个人资料
    • 访问:2394次
    • 积分:43
    • 等级:
    • 排名:千里之外
    • 原创:2篇
    • 转载:2篇
    • 译文:0篇
    • 评论:0条
    文章分类
    文章存档