C++动态链接库
1、动态链接库概述:
- 动态链接库通常都不能直接运行,也不能接受消息;只有在其他模块调用动态链接库中的函数时,它才发挥作用。
- Windows API中所有的函数都包含在动态链接库中。
- 动态链接库分静态库和动态库。
2、导出函数的声明方式
1)强制用C语言方式进行修饰,且用C的默认调用约定,即__cdecl方式。这种方式编译产生的DLL中有一个导出函数:add,不加任何修饰。(这种方式最好)
extern "C" int __declspec(dllexport) add();
2)强制用C语言方式进行修饰,且用__stdcall约定。这种方式编译产生的DLL中有一个导出函数:_add@0,即前面有“_”,后面加了参数长。
extern "C" int __declspec(dllexport) __stdcall add();
3)不强制用C语言方式进行修饰,但是用__stdcall约定。这种方式编译产生的DLL中有一个导出函数:?add@@YGHXZ。这个名字很怪,后面的不好理解。
int __declspec(dllexport) __stdcall add();
4)不强制用C语言方式进行修饰,并且用 __cdecl 约定。这种方式编译产生的DLL中有一个导出函数:?add@@YAHXZ。注意看,和第三种方有一点不同。
int __declspec(dllexport) __cdecl add();
对于DLL导出函数声明的四种写法,在动态调用时, 声明成第一种方式是最好的。但是,C/C++缺省的调用约定为__cdecl约定,如果想别的语言能用DLL的话,最好是将调用约定写成__stdcall方式(不能动态调用),然后静态(隐式)调用。
在隐式调用时,四种声明方式都是可以的,只要调用者的声明方式和DLL声明时的方式一致即可。
3、动态链接库的创建:
我们以第一种声明方式对函数进行声明,要导出的函数的形式为:
extern "C" int _declspec(dllexport) add()
{ return 5; }
为解决名字改编问题,可为工程添加一个模块定义文件.def
LIBRARY MFC-Add EXPORTS add@0 ;函数名@参数长
要导出的类的形式为:
class __declspec(dllexport) Point { public: void OutPut(int x, int y); };
编译之后会生成:一个DLL(.dll)文件、一个引入库(.lib)文件等
注:若在声明类时,指定了导出标志,那么该类中的所有函数均被导出;否则只导出那些声明时指定了导出标志的类成员函数
4、动态链接库的加载:
(1) 隐式加载
利用 extern 或 __declspec(dllimport) 声明外部函数(最好将此声明写在一个头文件*.h中)
extern "C" int _declspec(dllimport) add();
导入对应的引入库(.lib)文件:在项目“设置”中链接lib库 或 在源文件中使用代码链接lib库:
#pragma comment(lib, "*.lib")
将动态链接库文件所在目录添加到系统的环境变量path中。
附:Qt中库的隐式加载:
VC的引用库文件为xxx.lib, GCC的为xxx.a,通过比较两种库文件的格式,发现很相似,于是把xxx.dll,xxx.lib和xxx.h复制到Qt的project下,直接把xxx.lib改为xxx.a, 根据Qt的库名字的格式, 在xxx.a的前面加上lib, 即为libxxx.a。再在Qt的.pro文件中最后面加上
LIBS += -L. –lxxx //增加当前目录下的libxxx.a
先在pro文件中加入一行,LIBS += -L ./ -lPlotDll,然后将PlotDll.dll和PlotDll.lib拷贝到工程build目录下(就是含有Makefile的目录),再重新编译一次即可。
(2) 显式加载
将指定的可执行模块映射到调用进程的地址空间
#include <Windows.h> HINSTANCE handlerDLL = NULL; handlerDLL = LoadLibrary(_T("..\\..\\MFC-Add\\Debug\\MFC-Add.dll"));
获取该动态链接库中导出函数的地址
typedef int(__stdcall *ADDPROC)(int a,int b);//声明ADDPROC为指向函数的指针类型,函数返回值类型为int; _stdcall即标准调用约定 ADDPROC AddProc = NULL; AddProc = (ADDPROC)GetProcAddress(handlerDLL,_T("add"));
调用导出函数
AddProc(5,6);
当不需要访问dll时,释放对dll的引用
FreeLibrary("..\\..\\MFC-Add\\Debug\\MFC-Add.dll");
Qt中的动态调用方式,代码:
//动态加载 QLibrary hDll(".\\Libs\\GHC02DLL.dll"); if(hDll.load()) { typedef int (__stdcall *ADDFunc)(); ADDFunc Add=(ADDFunc)hDll.resolve("fnGHC02DLL"); QMessageBox::about(this,"about",QString::number(Add(),10)); hDll.unload(); }
注:
- Dumpbin 和 Depends 工具
- 调用约定:__stdcall:标准调用约定即WINAPI调用约定,也就是pascal调用约定,非C调用约定
- 名字改编:C++编译器在生成DLL时,会对导出函数进行名字改编,且不同的编译器使用的改编规则不同
相关链接:http://www.cnblogs.com/DxSoft/archive/2011/04/22/2024686.html
高洪臣 (Gavin Gao)
cggos@outlook.com
=======================================================================