Dll高级技术之【延迟加载】

前端之家收集整理的这篇文章主要介绍了Dll高级技术之【延迟加载】前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
【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;
}

猜你在找的Windows相关文章