显试调用DLL------GetProcAddress的使用

本文解释了typedef在定义函数指针类型中的应用,并详细介绍了如何使用GetProcAddress函数从DLL中获取函数地址。此外,还提供了一个完整的示例代码,演示了如何加载和调用DLL中的函数。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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 是两个函数指针类型。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值