【DLL的隐式加载】: 1.在Lib中包含导入库MyLib.lib 2.cpp中包含头文件MyLib.h 3.运行目录里面包含MyLib.dll 【DLL的显式加载】: 1.运行目录里面包含MyLib.dll 2.LoadLibrary及GetProcAddress 【DLL的延迟加载】: 1.需要DLL,MyLib.lib导入库及MyLib.h 进行【隐式加载】的全步骤 2.属性->配置属性->链接器->输入->延迟加载的Dll-> 添加MyDll.dll(注意/DelayLoad:MyDll.dll这个开关不能用#pragma comment(linker,"/DelayLoad:MyDll.dll")来设置。 3.属性->配置属性->链接器->输入->附加依赖项-> 添加DelayImp.lib //也可以用#include <delayimp.h>和#pragma comment(lib,"Delayimp.lib") //这个开关告诉链接器将delayimp中的__delayLoadHelper2函数嵌入到我们的可执行文件中。 4.如果需要自动卸载Dll,则需在可选 属性->配置属性->链接器->高级->卸载延迟加载的DLL->是 (/DELAY:UNLOAD); //此时只能调用__FUnloadDelayLoadedDll2(PCSTR szDll)函数,而不能调用FreeLibrary //并且传入的参数不包含路径,且名称与延迟加载的Dll中配置的参数必须保持一致,如果不打算卸载,就可以不指定/DELAY:UNLOAD =》好处1.可以在使用多个DLL初始化比较慢的时候,使用这一技术,可以将DLL的载入过程延迟到进程的执行过程中。 =》好处2.当一个函数在老版系统中不存在时,比如GerVersionEx在Windows Vista之前的系统没有此函数,在运行包含了这个函数的老系统中就会出错, =》好处3.减小编写LoadLibrary,GetProcAddress 而像静态库函数一样直接使用. ★延迟加载其实是动态库的隐示加载和显示加载的合并 ★延迟载入是针对隐式链接DLL的 ★一个导出了字段(如全局变量)的DLL是无法延迟载入的 ★Kernel32.dll模块是无法延迟载入的,因为必须载入该模块才能调用LoadLibrary和GetProcAddress。 ★不应在DllMain入口函数中调用一个延迟载入的函数,这可能导致程序崩溃
下面是示例程序:
#include "stdafx.h"
#include <windows.h>
#include <StrSafe.h>
//隐式链接DLL
#include "MyDll.h"
#pragma comment(lib,"..\\debug\\MyDll.lib")
//延迟加载DLL的名字
TCHAR g_szDelayLoadModuleName[] = TEXT("MyDll");
//延迟加载DLL的异常捕获
LONG WINAPI DelayLoadDllExceptionFilter(PEXCEPTION_POINTERS pep);
//检查是否加载
void IsModuleLoaded(PCTSTR pszModuleName)
{
HMODULE hmod = GetModuleHandle(pszModuleName);
char sz[100] = {0};
StringCchPrintfA(sz,_countof(sz),"MyDll.dll is %s loaded.",(hmod == NULL) ? L"not" : L"");
MessageBoxA(NULL,sz,0);
}
//延迟加载及异常捕获演示:
int main()
{
__try
{
IsModuleLoaded(g_szDelayLoadModuleName);
int sum = fnMyDll(2,43);//调用MyDll.dll中导出的API
IsModuleLoaded(g_szDelayLoadModuleName);
}
__except (DelayLoadDllExceptionFilter(GetExceptioninformation()))
{
}
// we can do otherthing ...
}
LONG WINAPI DelayLoadDllExceptionFilter(PEXCEPTION_POINTERS pep)
{
return -1;
}