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 和#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
#include
//隐式链接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,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;
}