http://support.microsoft.com/kb/815065/zh-cn
什么是 DLL?
本页概要
本文说明什么是动态链接库 (DLL) 以及在使用 DLL 时可能发生的各种问题。
然后,本文说明在开发您自己的 DLL 时应该考虑的一些高级问题。在说明什么是 DLL 的过程中,本文将说明动态链接方法、DLL 依赖性、DLL 入口点、导出 DLL 函数以及 DLL 故障排除工具。 本文最后将从较高的层次对 DLL 与 Microsoft .NET Framework 程序集作一比较。 简介
对于“适用于”一节中列出的 Microsoft Windows 操作系统,操作系统的大量功能是由动态链接库 (DLL) 提供的。另外,当您在这些 Windows 操作系统之一上运行某一程序时,该程序的很多功能可能是由 DLL 提供的。例如,某些程序可能包含很多不同的模块,而该程序的每个模块都包含在 DLL 中并从中分发。
使用 DLL 有助于促进代码的模块化、代码重用、内存的有效使用和减少所占用的磁盘空间。因此,操作系统和程序能够更快地加载和运行,并且在计算机中占用较少的磁盘空间。 当程序使用 DLL 时,一个称为依赖性的问题可能导致该程序无法运行。当程序使用 DLL 时,就会创建一个依赖项。如果其他程序改写和损坏了该依赖项,原来的那个程序就可能无法成功运行。 在引入 Microsoft .NET Framework 之后,大多数依赖性问题都已经通过使用程序集消除了。 更多信息什么是 DLL?DLL 是一个包含可由多个程序同时使用的代码和数据的库。例如,在 Windows 操作系统中,Comdlg32 DLL 执行与对话框有关的常见函数。因此,每个程序都可以使用该 DLL 中包含的功能来实现“打开”对话框。这有助于促进代码重用和内存的有效使用。通过使用 DLL,程序可以实现模块化,由相对独立的组件组成。例如,一个计帐程序可以按模块来销售。可以在运行时将各个模块加载到主程序中(如果安装了相应模块)。因为模块是彼此独立的,所以程序的加载速度更快,而且模块只在相应的功能被请求时才加载。 此外,可以更为容易地将更新应用于各个模块,而不会影响该程序的其他部分。例如,您可能具有一个工资计算程序,而税率每年都会更改。当这些更改被隔离到 DLL 中以后,您无需重新生成或安装整个程序就可以应用更新。 下表说明了 Windows 操作系统中的一些作为 DLL 实现的文件:
DLL 的优点下表说明了当程序使用 DLL 时提供的一些优点:
DLL 依赖项当某个程序或 DLL 使用其他 DLL 中的 DLL 函数时,就会创建依赖项。因此,该程序就不再是独立的,并且如果该依赖项被损坏,该程序就可能遇到问题。例如,如果发生下列操作之一,则该程序可能无法运行:
下表说明了为了帮助最大限度地减少依赖性问题而在 Microsoft Windows 2000 和较高版本的 Windows 操作系统中引入的更改:
DLL 故障排除工具可以使用多个工具来帮助您解决 DLL 问题。以下是其中的部分工具。Dependency WalkerDependency Walker 工具可以递归扫描以寻找程序所使用的所有依赖 DLL。当您在 Dependency Walker 中打开程序时,Dependency Walker 会执行下列检查:
drive/Program Files/Microsoft Visual Studio/Common/Tools
DLL Universal Problem SolverDLL Universal Problem Solver (DUPS) 工具用于审核、比较、记录和显示 DLL 信息。下表说明了组成 DUPS 工具的实用工具:
247957 示例:使用 DUPS.exe 解决 DLL 兼容性问题
DLL 帮助数据库DLL 帮助数据库帮助您查找由 Microsoft 软件产品安装的特定版本的 DLL。有关 DLL 帮助数据库的更多信息,请访问下面的 Microsoft 网站:DLL 开发本节介绍您在开发自己的 DLL 时应该考虑的问题和要求。DLL 的类型当您在应用程序中加载 DLL 时,可以使用两种链接方法来调用导出的 DLL 函数。这两种链接方法是加载时动态链接和运行时动态链接。加载时动态链接在加载时动态链接中,应用程序像调用本地函数一样对导出的 DLL 函数进行显式调用。要使用加载时动态链接,请在编译和链接应用程序时提供头文件 (.h) 和导入库文件 (.lib)。当您这样做时,链接器将向系统提供加载 DLL 所需的信息,并在加载时解析导出的 DLL 函数的位置。运行时动态链接在运行时动态链接中,应用程序调用 LoadLibrary 函数或 LoadLibraryEx 函数以在运行时加载 DLL。成功加载 DLL 后,可以使用 GetProcAddress 函数获得要调用的导出的 DLL 函数的地址。在使用运行时动态链接时,无需使用导入库文件。下面的列表说明了有关何时使用加载时动态链接以及何时使用运行时动态链接的应用程序条件:
DLL 入口点在创建 DLL 时,可以有选择地指定入口点函数。当进程或线程将它们自身附加到 DLL 或者将它们自身从 DLL 分离时,将调用入口点函数。您可以使用入口点函数根据 DLL 的需要来初始化数据结构或者销毁数据结构。此外,如果应用程序是多线程的,则可以在入口点函数中使用线程本地存储 (TLS) 来分配各个线程专用的内存。下面的代码是一个 DLL 入口点函数的示例。
入口点函数只应执行简单的初始化任务,不应调用任何其他 DLL 加载函数或终止函数。例如,在入口点函数中,不应直接或间接调用 LoadLibrary 函数或 LoadLibraryEx 函数。此外,不应在进程终止时调用 FreeLibrary 函数。 注意:在多线程应用程序中,请确保将对 DLL 全局数据的访问进行同步(线程安全),以避免可能的数据损坏。为此,请使用 TLS 为各个线程提供唯一的数据。 导出 DLL 函数要导出 DLL 函数,您可以向导出的 DLL 函数中添加函数关键字,也可以创建模块定义文件 (.def) 以列出导出的 DLL 函数。要使用函数关键字,您必须使用以下关键字来声明要导出的各个函数:
__declspec(dllexport)
要在应用程序中使用导出的 DLL 函数,您必须使用以下关键字来声明要导入的各个函数:
__declspec(dllimport)
通常情况下,您最好使用一个包含
define 语句和
ifdef 语句的头文件,以便分隔导出语句和导入语句。
您还可以使用模块定义文件来声明导出的 DLL 函数。当您使用模块定义文件时,您不必向导出的 DLL 函数中添加函数关键字。在模块定义文件中,您可以声明 DLL 的 LIBRARY 语句和 EXPORTS 语句。下面的代码是一个定义文件的示例。
示例 DLL 和应用程序在 Microsoft Visual C++ 6.0 中,可以通过选择“Win32 动态链接库”项目类型或“MFC 应用程序向导 (dll)”来创建 DLL。下面的代码是一个在 Visual C++ 中通过使用“Win32 动态链接库”项目类型创建的 DLL 的示例。
在运行时动态链接中,您应使用与以下代码类似的代码来调用 SampleDLL.dll 导出 DLL 函数。
.NET Framework 程序集在引入 Microsoft .NET 和 .NET Framework 以后,大多数与 DLL 相关联的问题已经通过使用程序集消除了。程序集是在 .NET 公共语言运行库 (CLR) 控制之下运行的逻辑功能单元。程序集实际上是作为 .dll 文件或 .exe 文件存在的。但是,在内部,程序集与 Microsoft Win32 DLL 大不相同。程序集文件包含程序集清单、类型元数据、Microsoft 中间语言 (MSIL) 代码和其他资源。程序集清单包含程序集元数据,以提供使程序集成为自描述程序集所需的全部信息。程序集清单中包含以下信息:
下表说明了程序集的一些功能,并将其与 Win32 DLL 的功能进行了比较:
参考
有关 DLL 和 .NET Framework 程序集的更多信息,请访问下面的 Microsoft 网站:
DLL 帮助数据库
http://support.microsoft.com/dllhelp DLL 冲突 (DLL conflicts) http://msdn2.microsoft.com/en-us/library/ms811694.aspx 在应用程序中实现并行组件共享 (Implementing side-by-side component sharing in applications) http://msdn2.microsoft.com/en-us/library/ms811700.aspx 如何生成和维护用于 Windows XP 的独立应用程序和并行程序集 (How to build and service isolated applications and side-by-side assemblies for Windows XP) http://msdn2.microsoft.com/en-us/library/ms997620.aspx 使用 .NET Framework 简化部署和解决 DLL 冲突 (Simplifying deployment and solving DLL conflicts with the .NET Framework) http://msdn2.microsoft.com/en-us/netframework/aa497268.aspx .NET Framework 开发人员指南:程序集 http://msdn2.microsoft.com/en-us/library/hk5f40ct(vs.71).aspx 运行时动态链接 http://msdn2.microsoft.com/en-us/library/ms685090.aspx 线程本地存储 (Thread local storage) http://msdn2.microsoft.com/en-us/library/ms686749.aspx |