您的位置:首页 > 其它

动态链接库中导出函数的调用

2017-03-27 10:56 344 查看
动态链接库中的函数导出成功后,其它EXE程序要使用导出函数,就会涉及到如何调用动态链接库。

调用DLL也分为两种方法:隐式调用和显示调用:

1、隐式调用:

隐式链接的特点是由编译器完成对DLL的加载和程序结束时对DLL的卸载工作,如果程序结束时如果还有其他应用程序使用该DLL,那么系统会使DLL的使用计数减1,当DLL的使用计数降为0时,会将DLL从内存中删除。

★优缺点:隐式链接DLL的方法简单实用,但缺少灵活性。

■使用方法:使用隐式链接DLL开发时,首先,需要将DLL的引入库文件(*.lib--编译生成DLL时,会一起生成的)与应用程序进行静态链接,因为引入库文件包含DLL的各种输出资源,如导出函数,导出类等信息,这些信息指向DLL的函数指针等等,EXE执行时,DLL被 “自动” 加载,EXE退出时DLL被 “自动” 卸载。

对上节所讲的Add()函数进行隐式调用:

#pragma comment(lib,"D:\\DllTest\\Debug\\DllTest.lib")
_declspec(dllimport) int Add(int a,int b);


2、显示调用:

显示链接方式是完全由编程者用API 加载和卸载DLL,编程者可以决定何时加载DLL,加载哪个DLL,何时卸载DLL,卸载哪个DLL等。

★优缺点:显示链接方式充分体现了DLL的灵活性,是比较常用的调用DLL方式。但是与静态链接相比稍微复杂了些。

■使用方法:

LoadLibrary(...):该 API 用于加载指定的DLL;

GetProcAddress(...):该 API 用于获取DLL中导出函数的指针, 即导出函数的入口点;

FreeLibrary(...):该 API 用于卸载指定的DLL。

〓注:如果程序中多次调用LoadLibrary(...)加载同一DLL时,在卸载的时候也要调用相应次数的FreeLibrary(...)进行卸载。

对上节所讲的Dec()函数进行显示调用:

typedef int (*MYPROC)(int a,int b);

HMODULE hMod = LoadLibrary("D:\\DllTest\\Debug\\DllTest.dll");//加载指定DLL模块
if (hMod)
{
MYPROC Func = (MYPROC)GetProcAddress(hMod,"Dec");//获取DLL中导出函数的指针,即导出函数的入口点
if (Func)
{
nRet=Func(10,6);
printf("10-6=%d\n",nRet);
}
FreeLibrary(hMod);//卸载指定的DLL
}


▲使用LoadLibrary显式链接,在这个函数的参数中可以指定DLL文件的完整路径。如果不指定路径,Windows将遵循如下的搜索顺序来定位DLL:

①EXE文件所在的目录:

②进程的当前工作目录:

③Windows系统目录:例如:C:\WINDOWS\system32

④Windows目录:例如:C:\WINDOWS

⑤环境变量的目录:我的电脑-->属性-->高级-->环境变量

/****************************************************************************/

函数指针:

对于一个函数:int Add(int a ,int b);

定义一个函数指针:    int (*pFunc)(int a ,int b);  pFunc为函数指针

定义一个函数指针类型:typedef int (*pFunc)(int a,int b); pFunc为函数指针类型

                      pFunc  p; p为函数指针
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: