天天看點

顯試調用DLL------GetProcAddress的使用

1、關于typedef int(*lpAddFun)(int, int)  

在非MFC DLL學習中出現了一個不太好懂的語句:

typedef int(*lpAddFun)(int, int); //宏定義函數指針類型 
      
這句宏定義函數指針類型是什麼意思呢? 資料來源:yuhaibin echo      
我們知道typedef 是宏定義,一般的文法是這樣:      
typedef unsigned long ulong; 
      
這個定義是用ulong作unsigned long 的别名      
而在這裡的意思是定義一種指針類型lpAddFun,它是一種指向函數int (int,int)的指針,也就是說lpAddFun表示的是這種類型的函數的位址。      
換句話說,lpAddFun是typedef定義的一個名稱,可以用來定義變量。      
比如  lpAddFun p;      
那p就是  int(*p)(int, int);      
 首先(*p)說明p是一個指針,(*p)();說明p指向函數,      
(*p)(int, int)說明p指向的函數有兩個int類型參數,      
最後  int(*p)(int, int);說明p指向的函數傳回值類型是int。      
2、
 
          

功能描述

編輯 函數原型: FARPROC GetProcAddress( HMODULE hModule, // DLL子產品句柄 LPCSTR lpProcName // 函數名 );

參數

編輯 hModule [in] 包含此函數的DLL子產品的句柄。LoadLibrary、 AfxLoadLibrary 或者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。

參看

編輯 動态連結庫縱覽( Dynamic-Link Libraries Overview), 動态連結庫函數(Dynamic-Link Library Functions),FreeLibrary, GetModuleHandle, LoadLibrary 示例代碼: 調用KERNEL32.DLL中的RegisterServiceProcess(僅在Windows98中适用) HMODULE hModule=GetModuleHandle("kernel32.dll"); if (hModule)   {   typedef DWORD (CALLBACK *LPFNREGISTER)(DWORD,DWORD);   LPFNREGISTER lpfnRegister;   lpfnRegister=(LPFNREGISTER)GetProcAddress(hModule,"RegisterServiceProcess");   if (lpfnRegister)   {   (*lpfnRegister)(NULL,1L);   }   }
3、舉個例子說明:和調用動态庫時的那個typedef一緻      
#include <stdio.h>

#include <windows.h>

 

typedef int (*ADD)(int, int);

typedef int (*SUB)(int, int);

 

int main(int argc, char* argv[])

{

         HMODULE hmodule11 = ::LoadLibrary("DynamicDLL.dll");

         ADD Add= (ADD) GetProcAddress(hmodule11,"GetAdd");

         SUB Sub= (SUB) GetProcAddress(hmodule11,"GetSub");

 

          int x=10,y=5;

         int nRes = Add(x,y);

         int nRes2= Sub(x,y);

 

         FreeLibrary(hmodule11);

         printf("%d,%d\n",nRes,nRes2);

         return 0;

}
      
〉其中ADD 和SUB 是兩個函數指針類型。      

繼續閱讀