为了让线程能够调用DLL中的函数,我们必须把DLL中的文件映像映射到调用线程所在进程的地址空间中,有两种方法实现:
第一种是直接在源代码中引用DLL中所包含的函数或是变量,DLL在程序运行后由加载程序隐式地载入,此种方式被称为隐式链接。
第二种方式是在程序运行时,通过调用LoadLibrary显式载入所需要的DLL,并显式地链接想要的符号。通过GetProcAddress函数可以获得显式载入的DLL中符号的地址。
显式加载后,exe的数据目录中并没有导入表,可以通过相关的函数获取目标函数的地址,然后调用。
隐式加载有导入表,不需要程序员自己去获取目标函数的地址,直接调用即可。
隐式加载DLL后,在我们调用需要的DLL中的函数时,那么这个获得函数地址过程是怎样的呢?
在exe程序运行的时候,windows的加载程序会很据调用的函数找到其所在的DLL,然后将其导入到进程的地址空间中,导入程序后获得DLL中的函数然后将每一个函数放到导入表中DLL目录下,在调用某个函数的时候,那么就在导入表中找到其位置,就知道了这个函数的地址,调用即可。
显式加载DLL,那么如何得到DLL中的某个函数的地址呢?
显式加载DLL在进程的地址空间中是没有导入表的,我们可以通过两个函数来获得指定的函数的地址。这两个API分别是GetModuleHandle和GetProcAddress。
(1)HMODULEGetModuleHandle(LPCTSTRlpModuleName);
(2)函数原型:
{
typedef DWORD (CALLBACK *LPFNREGISTER)(DWORD,DWORD);
LPFNREGISTER lpfnRegister;
lpfnRegister=(LPFNREGISTER)GetProcAddress(hModule,"RegisterServiceProcess");
if (lpfnRegister)
{
(*lpfnRegister)(NULL,1L);
}
}