def模块定义方式创建动态链接库与动态加载使用dll
前面介绍了通过宏定义_declspec(dllexport)的方式创建动态链接库,需要定义头文件和宏,定义函数时需要使用宏定义,这种方式过于繁琐,这里介绍使用模块定义文件的方式创建动态链接库,只需要一个源文件和一个def文件即可,代码非常简单,通过动态加载的方式使用dll也非常简单,只需要将动态链接库放入项目路径下即可。
1、创建动态链接库工程,并手工新建文件Dll1.def,定义需要导出的方法名和动态链接库名。
Dll1.def文件内容如下:
LIBRARY Dll1 EXPORTS add substract printHello
这里我们增加了printHello方法,调用c++的输出,向控制台打印Hello,JNA。
2、创建源文件,并编码。
#include "stdafx.h" #include <iostream> using namespace std; int add(int a, int b) { return a + b; } int substract(int a, int b) { return a - b; } void printHello() { cout << "Hello,JNA" << endl; }
3、这里需要设置一下,将Dll1.def文件和项目关联起来。执行项目右键->属性->链接器->输入->右侧模块定义文件->选择项目中的Dll1.def文件。如下图:
4、生成,这里如果不将模块定义文件指定到项目中,生成时不会报错,但是生成的dll文件中不会导出我们需要给客户端使用的函数,违背了我们的初衷。
5、dumpbin查看Dll1.dll文件,验证导出函数。
结果证明,导出函数成功,并且方法名没有发生变化。
6、通过动态加载的方式使用动态链接库。还是创建控制台空项目。并将Dll1.dll文件加入项目路径下。然后新建源文件,编码。
这里使用动态加载的方式,就需要手动编写获取dll导出函数的方法,并调用。
#include <iostream> #include "Windows.h" using namespace std; int main() { HMODULE hDll; hDll = LoadLibrary(L"Dll1.dll"); typedef int(*FUNCTION)(int a, int b); FUNCTION ADD = (FUNCTION)GetProcAddress(hDll, "add"); FUNCTION SUB = (FUNCTION)GetProcAddress(hDll, "substract"); if (!ADD) { cout << "获取地址失败" << endl; return 0; } cout << "dynamic load add(5,2)=" << ADD(5, 2) << endl; cout << "dynamic load substract(5,2)=" << SUB(5, 2) << endl; return 0; }
与静态链接的方式使用dll相比,动态加载的方式,不用将lib文件也加入到项目中,只需要将dll文件拷贝到项目路径下,然后编写调用过程时,因为是动态加载,所以每次使用的时候,必须通过获取函数地址的方式来获取函数名,最终调用函数。
运行结果如下: