函數功能描述:GetProcAddress函數檢索指定的動態鏈接庫(DLL)中的輸出庫函數地址。(使用動態調用函數地址之后,第三方程序(比如Depend.exe)就不會查出程序使用了哪些庫和哪些函數)
函數原型:
FARPROC GetProcAddress(
HMODULE hModule, // DLL模塊句柄
LPCSTR lpProcName // 函數名
);
參數:
hModule
[in] 包含此函數的DLL模塊的句柄。LoadLibrary或者GetModuleHandle函數可以返回此句柄。
lpProcName
[in] 包含函數名的以NULL結尾的字符串,或者指定函數的序數值。如果此參數是一個序數值,它必須在一個字的底字節,高字節必須為0。
返回值:
如果函數調用成功,返回值是DLL中的輸出函數地址。
如果函數調用失敗,返回值是NULL。得到進一步的錯誤信息,調用函數GetLastError。
注釋:
GetProcAddress函數被用來檢索在DLL中的輸出函數地址。
lpProcName指針指向的函數名,拼寫和大小寫必須和DLL源代碼中的模塊定義文件(.DEF)中輸出段(EXPORTS)中指定的相同。Win32 API函數的輸出名可能不同於你在代碼中調用的這些函數名,這個不同被宏隱含在相關的SDK頭文件中。如果想得到更多信息,請參考Win32函數原型(Win32 Function Prototypes)。
lpProcName參數能夠識別DLL中的函數,通過指定一個與函數相聯系的序數值(在.DEF中的EXPORTS段)。GetProcAddress函數驗證那個指定的序數值是否在輸出的序數1和最高序數值之間(在.DEF中)。函數用這個序數值作為索引從函數表中讀函數地址,假如.DEF 文件不連續地定義函數的序數值,如從1到N(N是輸出的函數序數值),錯誤將會發生,GetProcAddress將會返回一個錯誤的、非空的地址,雖然指定的序數沒有對應的函數。
為了防止函數不存在,函數應該通過名字指定而不是序數值。
要求:
Windows NT/2000: 要求Windows NT 3.1 或以后版本。
Windows 95/98: 要求Windows 95 或以后版本。
頭文件: 在Winbase.h中聲明,include Windows.h。
庫文件: Use Kernel32.lib。
注意問題:
1、GerProcAddress要導出的函數A地址, A必須真實存在,而不能是宏。例如:要導出MessageBox則程序會崩潰,因為MessageBox不是一個真實的函數,它只是一個宏,MessageBoxA和MessageBoxW才是真實的函數,所以導出函數前先查閱一下MSDN,看函數介紹最后是否有下面的一句話,如果存在則表明該函數有ANSI版本和Unicode版本的區別。
2、用該函數導入dll里面的函數地址后,最好判斷函數指針是否為空,檢查是否導入成功,否則程序會崩潰的。
Requirements
// Windows NT/2000: Requires Windows NT 3.1 or later.
// Windows 95/98: Requires Windows 95 or later.
// Header: Declared in Winuser.h; include Windows.h.
// Library: Use User32.lib.
Unicode: Implemented as Unicode and ANSI versions on all platforms.
另外,需要注意的是,在動態調用函數的時候,寫函數指針宏定義的時候,最好轉到文件具體定義中去,盡量不要用msdn中的。例如:
// WinReg.h文檔中的定義
WINADVAPI
LSTATUS
APIENTRY
RegOpenKeyExA (
__in HKEY hKey,
__in_opt LPCSTR lpSubKey,
__reserved DWORD ulOptions,
__in REGSAM samDesired,
__out PHKEY phkResult
);
WINADVAPI
LSTATUS
APIENTRY
RegOpenKeyExW (
__in HKEY hKey,
__in_opt LPCWSTR lpSubKey,
__reserved DWORD ulOptions,
__in REGSAM samDesired,
__out PHKEY phkResult
);
#ifdef UNICODE
#define RegOpenKeyEx RegOpenKeyExW
#else
#define RegOpenKeyEx RegOpenKeyExA
#endif // !UNICODE
// MSDN中的定義
LONG WINAPI RegOpenKeyEx(
__in HKEY hKey,
__in LPCTSTR lpSubKey,
DWORD ulOptions,
__in REGSAM samDesired,
__out PHKEY phkResult
);
不管使用ANSI編碼,還是UNICODE編碼 都用文檔中定義那個,要不然在參數傳遞的是可能會報錯參數類型不對。另外建議也不要用TCHAR,要么char,要么wchar_t,因為你在代碼編寫的時候,涉及到編碼的時候不清楚某個變量是什麽類型了,那樣子很麻煩。一個函數中都用char或wchar_t,即使參數傳遞的時候可以用ANSI和UNICODE轉換。
【參考資料 感謝作者】
GetProcAddress函數