动态库dll的调用方式-显式链接

这里有个疑问,VS2017生成dll时为何会同时生成一个lib文件?

静态库生成lib,动态库也生成lib。

 

动态库生成的那个lib一般管它叫"导入库",这样的lib在大多数情况下要比静态库的lib小,
里面不包含涉及到的函数的具体代码,里面只包含"这个函数在什么dll里面叫什么名字"这样的信息。即函数的声明
你平时编程比如windows.h里的那些api,都是通过这种方式链接的。
比如弹出消息框的MessageBoxA,它在user32.lib中,但是user32.lib却没有包含MessageBoxA的具体代码,
它只写明了这个函数在user32.dll中,名字叫做MessageBoxA。生成exe以后,windows在运行这个exe里的代码之前,
会先载入user32.dll,然后在里面查找MessageBoxA,随后把MessageBoxA所在的地址写入exe指定的位置(一般是导入表)。
dll要调用这个函数的时候,就从导入表获取函数地址,然后调用过去。

 

 

方法2:显式链接---需要DLL(不需要.lib,.h文件)

注意:显示链接,导出DLL的时候,采用extern “C”的方式

而不采用_declspec(dllexport),因为_declspec(dllexport)会有一个”名字改编”的问题(采用了_cdecl调用规约的C++编译方式)。

 

如库文件对应的头文件如下

#ifndef DLLTEST_H

#defineDLLTEST_H

 

//该宏完成在dll项目内部使用__declspec(dllexport)导出

//在dll项目外部使用时,用__declspec(dllimport)导入

//宏DLL_EXPORTS在.cpp中定义

#ifdefDLL_EXPORTS

#defineDLL_EXPORTS extern"C"_declspec(dllexport)

#else

#define DLL_EXPORTS extern"C"_declspec(dllimport)

#endif

//函数声明

DLL_EXPORTSint  Add(int a,int b);

DLL_EXPORTSint  Sub(int a,int b);

DLL_EXPORTSint  Divide(int a,int b);

#endif//DLLTEST_H

 

 

测试代码:

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

      typedef int(fun *Sub)(int a, int b);

      HINSTANCE hDLL;

      funAdd Add;//函数指针

      hDLL =LoadLibrary(_T("D:/DLLTest.dll"));//加载动态链接库DLLTest.dll文件;

      Add = (funAdd)GetProcAddress(hDLL,"Add");

      int result = Add(5, 8);

      printf("5+8:%d\n",result);

      FreeLibrary(hDLL);//卸载.dll文件;
以下是采用__declspec(dllexport)方式导出,不推荐这种方式需要注意函数名的书写问题!

现在DLLTest.h中有函数(采用__declspec(dllexport)方式导出)

#ifndef DLLTEST_H

#defineDLLTEST_H

 

//该宏完成在dll项目内部使用__declspec(dllexport)导出

//在dll项目外部使用时,用__declspec(dllimport)导入

//宏DLL_EXPORTS在.cpp中定义

#ifdefDLL_EXPORTS

#defineDLL_EXPORTS __declspec(dllexport)

#else

#define DLL_EXPORTS __declspec(dllimport)

#endif

 

 

intDLL_EXPORTS Add(int a,int b);

intDLL_EXPORTS Sub(int a,int b);

intDLL_EXPORTS Divide(int a,int b);

 

#endif//DLLTEST_H
测试代码

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

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

      HINSTANCE hDLL;

      funAdd Add;//函数指针

      //也可以采用hDLL =LoadLibrary(_T("D:/DLLTest.dll"));

      hDLL =LoadLibraryA(("D:/DLLTest.dll"));//加载动态链接库DLLTest.dll文件;

    Add = (funAdd)GetProcAddress(hDLL,"?Add@@YAHHH@Z");//!!!!获取函数地址

      int result = Add_(5, 8);

      printf("5+8:%d\n",result);

      FreeLibrary(hDLL);//卸载.dll文件;

这里需要注意的是:GetProcAddress(hDLL,"函数名");中的函数名要是DLL中的函数名,这个函数名可以用PE Explorer软件查看

由于采用的是VC++处理函数名方式,所以

"?Add@@YAHHH@Z";// GetProcAddress(hDLL,"函数名");

 

而不是简单的“Add”,因为DLL中的函数名是结果VC++方式处理过的函数名

 

显式调用,最好采用extern “C”的方式导出DLL

 

所以显式隐式链接方式的时候,只加载需要的DLL在附加依赖项中,只添加需要的DLL 对于的lib,不要多加

否则会造成

1.加大程序启动时间

2.内存浪费

 

posted @ 2021-08-11 21:17  磐正  阅读(660)  评论(0)    收藏  举报