Learn DLL
8. DLL延迟加载
DLL延迟加载是VC提供的一项特性,主要是在隐式链接时并不实际加载DLL,而是在要引用DLL的到处符号时才加载的一项技术。它在以下方面比较有用:
(1) 当程序引用多个DLL时,在初始化阶段使用该技术可以减少初始化时间,提高效率
(2) 当程序引用了系统提供的新函数,而程序运行在旧版本系统(未提供该函数)时,通常系统会因初始化失败而无法运行;使用延迟加载技术以后程序可以启动并运行,这样我们就有机会去在运行时决定是否调用新函数。
但是该项技术对能够延迟加载的DLL有以下限制:
(1) DLL不能导出字段(Field)
(2) kernel32.dll不能延迟加载
(3) 不能在dllMain函数中调用延迟加载的dll导出函数
如何在程序中实现延迟加载某个DLL呢?需要在工程linker属性中添加如下开关:
· /Lib:DelayImp.lib
· /DelayLoad:MyDll.dll
注意,不可以在源码中使用#pragma comment(linker,””)指令实现此功能。
9. dll函数转发器
DLL函数转发器是DLL导出节中项,每一项把一个函数调用重定向到另一个DLL中的某个函数。
在DLL中可以这样实现函数转发:
#pragma comment(linker, "/export:SomeFunc=DllWork.SomeOtherFunc")
10. Known DLLs
某些OS提供的DLLs得到了特别对待,它们称为known DLLs。OS总是到同一个目录下去查找并加载这些DLL。这些DLL的注册表入口为:
HKEY_LOCAL_MACHINE/SYSTEM/CurrentControlSet/Control/Session Manager/KnownDLLs
假设我们在该注册表入口中添加以下值
value name: SomeLib
value data: Other.dll
如果我们想使用LoadLibrary加载 SomeLib.dll,那么你必须这样调用 LoadLibrary(“SomeLib”)
如果使用LoadLibrary(“SomeLib.dll”), 则实际加载的是Other.dll. 原因在于搜索的顺序不同。
如果不加dll扩展名,是前面说过的普通搜索顺序;如果加了扩展名,系统会先把扩展名去掉,然后搜索known dlls的注册表入口,如果找到就加载该dll,找不到才会按正常顺序去搜索。
11. DLL 重定向
在Windows刚开发的时候,RAM和Disk都比较小,价格昂贵;因此Windows尽量让资源共享。针对这个目的,微软建议多个应用共享的模块都放到系统目录,如C/C++运行时库,这使系统可以方便定位这些共享文件;但是后来随着技术的发展,RAM、Disk不再成为问题的时候,windows又建议应用把自己的库放到自己的目录,以免影响其它应用;同时不要动系统目录的东西。为达到此目的,windows从win2000开始添加了DLL重定向特性,特使系统在加载模块时首先搜索应用程序目录,只有在找不到的时候才会去搜索其它目录。
为强制windows先搜所应用程序目录,你需要创建一个appname.local文件;或者创建一个appname.local目录,把你的dll放到该目录下;这对于COM对象非常有用,即使不同的应用程序注册了同样的COM也不会出错。
12. 重设模块基地址
通常,在构建exe或者dll时,linker会为其设定首选基地址,exe为0x00400000, dll为0x10000000。当exe被调用时,系统会为新进程创建地址空间,并把exe模块映射到该基地址上;加载一个dll时,则把它映射到0x10000000,但是如果要调用多个dll那问题就来了,系统必须去访问dll的重定位节,修改符号的正确的地址。
如果exe/dll不能映射到它的首选基地址,有两个缺点:
(1)系统loader会去查看重定位节,修改许多代码,这会影响程序性能
(2)当loader修改模块的数据时,系统的写时复制机制又会强制把模块修改页进行copy,这意味着模块的页不能丢弃,也不能从模块映像重载,必须使用系统页文件进行换入换出,这非常影响系统性能。
另外,可以创建不包含重定位节的模块,如果不能把这样的模块映射到其首选基地址,则出错。
因此,如果要把多个模块映射到一个地址空间,最好为每一个模块都重设首选基地址,这可以在VC的工程linker属性中设置。