1.基本概念
?在应用程序中加载 DLL 时,可以使用两种链接方法来调用导出的 DLL 函数。这两种链接方法分别是加载时动态链接和运行时动态链接。
?1.1加载时动态链接
??加载时动态链接在加载时动态链接中,应用程序像调用本地函数一样对导出的 DLL 函数进行显式调用。要使用加载时动态链接,请在编译和链接应用程序时提供头文件 (.h) 和导入库文件 (.lib)。当这样做时,链接器将向系统提供加载 DLL 所需的信息,并在加载时解析导出的 DLL 函数的位置。
?1.2运行时动态链接
??运行时动态链接在运行时动态链接中,应用程序调用 LoadLibrary 函数或 LoadLibraryEx 函数以在运行时加载 DLL。成功加载 DLL 后,可以使用 GetProcAddress 函数获得要调用的导出的 DLL 函数的地址。在使用运行时动态链接时,无需使用导入库文件。
?1.3两种方式的区别
??如果应用程序的初始启动性能很重要,则应使用运行时动态链接。 易用性在加载时动态链接中,导出的 DLL 函数类似于本地函数。这使得可以方便地调用这些函数。应用程序逻辑在运行时动态链接中,应用程序可以分支,以便按照需要加载不同的模块。在开发多语言版本时,这一点很重要。DLL 入口点在创建 DLL 时,可以有选择地指定入口点函数。当进程或线程将它们自身附加到 DLL 或者将它们自身从 DLL 分离时,将调用入口点函数。可以使用入口点函数根据 DLL 的需要来初始化数据结构或者销毁数据结构。
?1.4注意事项
??(1).入口点函数只应执行简单的初始化任务,不应调用任何其他 DLL 加载函数或终止函数。例如,在入口点函数中,不应直接或间接调用 LoadLibrary 函数或 LoadLibraryEx 函数。此外,不应在进程终止时调用 FreeLibrary 函数。此外,当入口点函数返回 FALSE 值时,如果您使用的是加载时动态链接,则应用程序不启动。如果使用的是运行时动态链接,则只有个别 DLL 不会加载。 ??(2).如果应用程序是多线程的,则可以在入口点函数中使用线程本地存储 (TLS) 来分配各个线程专用的内存。在多线程应用程序中,请确保将对 DLL 全局数据的访问进行同步(线程安全),以避免可能的数据损坏。为此,请使用 TLS 为各个线程提供唯一的数据。 想要导出 DLL 函数,可以在DLL文件中用关键字__declspec(dllexport)声明要导出的函数。要在应用程序中使用导出的 DLL 函数,必须使用关键字__declspec(dllimport)来声明要导入的函数。通常情况下,最好使用一个包含 define 语句和 ifdef 语句的头文件,以便分隔导出语句和导入语句。 ??(3).还可以使用模块定义文件来声明导出的 DLL 函数。当您使用模块定义文件时,您不必向导出的 DLL 函数中添加函数关键字。在模块定义文件中,您可以声明 DLL 的 LIBRARY 语句和 EXPORTS 语句。下面的代码是一个定义文件的示例。 // SampleDLL.def//LIBRARY "sampleDLL"EXPORTS HelloWorld ??(4).当编译和链接应用程序时,Windows 操作系统将按照以下顺序在下列位置中搜索DLL文件:应用程序文件夹、当前文件夹、Windows 系统文件夹(可以用GetSystemDirectory 函数来返回 Windows 系统文件夹的路径)。
?1.5本文示例DLL源码
extern "C" _declspec(dllexport) void sample_function();
void sample_function() {
MessageBoxA(NULL, "导出函数被调用成功", "信息", MB_OK);
}
BOOL APIENTRY DllMain( HMODULE 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:
break;
}
return TRUE;
}
??其中extern "C"是用来去除函数名修饰的,_declspec(dllexport)是一个关键字,用来声明导出函数,关于函数名修饰介绍请参照DLL文件_DLLMain函数详解,源码请参照其附件,此处只多添加了一个示例导出函数。
2.加载时动态链接
#include <stdio.h>
#include <Windows.h>
#pragma comment(lib,"Dll_Create")
extern "C" _declspec(dllimport) void sample_function();
int main() {
sample_function();
return 0;
}
?2.1#pragma comment
|