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文件拷贝到项目路径下,然后编写调用过程时,因为是动态加载,所以每次使用的时候,必须通过获取函数地址的方式来获取函数名,最终调用函数。

运行结果如下:

猜你喜欢

转载自blog.csdn.net/feinifi/article/details/79838737