动态库的创建和使用(Win10+VS2022)

2022-07-10 09:59:46

在使用VS2022创建动态库的过程中,发现一个有趣的现象(或者说是奇怪的现象),就是创建的项目中不包含模块定义文件(.def文件)时,生成的结果中只有dll文件,没有lib文件;手动添加模块定义文件(.def文件)后,生成的结果中才包含了dll文件和lib文件。

动态库的创建
使用VS2022新建动态链接库项目,项目名为MyDll,项目所在文件夹为“E:\temp”。

在这里插入图片描述
在这里插入图片描述
点击创建,即完成项目的创建。

在VS2022的“解决方案资源管理器”视图中,添加名为“MyDllFunc.h”(文件名可以任意)的头文件(右键单击“头文件-添加-新建项-头文件”),输入代码如下:

#ifndef _MY_DLL_FUNC_H_#define _MY_DLL_FUNC_H_intmy_dll_add(int a,int b);#endif

在VS2022的“解决方案资源管理器”视图中,添加名为“MyDllFunc.cpp”(文件名可以任意)的C++文件(右键单击“源文件-添加-新建项-C++文件”),输入如下代码:

#include"pch.h"#include"MyDllFunc.h"intmy_dll_add(int a,int b){return a+ b;}

生成解决方案,即可生成库文件“E:\temp\MyDll\x64\Debug\MyDll.dll”。

动态库的使用-动态调用(只使用dll文件)
使用VS2022新建空项目,项目名为MyApp,项目所在文件夹为“E:\temp”。

在VS2022的“解决方案资源管理器”视图中,为项目MyApp添加名为“main.cpp”(文件名可以任意)的C++文件(右键单击“源文件-添加-新建项-C++文件”),输入如下代码:

#include <iostream>intmain(){return0;}

生成解决方案,会得到可执行文件“E:\temp\MyApp\x64\Debug\MyApp.exe”,将动态库文件“E:\temp\MyDll\x64\Debug\MyDll.dll”复制到exe文件所在目录。可使用如下命令

  copy E:\temp\MyDll\x64\Debug\MyDll.dll E:\temp\MyApp\x64\Debug

在项目属性页的高级选项中,将字符集改为“使用多字节字符集”;然后在main函数中添加加载动态库的代码,最终“main.cpp”文件的内容如下:

#include <Windows.h>#include <iostream>intmain(){HINSTANCE hDll=0;
	hDll=LoadLibrary("MyDll.dll");if(!hDll){
		std::cout<<"动态链接库 MyDll.dll 加载失败"<< std::endl;return-1;}typedefint(*PFN_ADD)(int a,int b);//	定义函数指针类型void*pfn=GetProcAddress(hDll,"my_dll_add");if(pfn==0){
		std::cout<<"动态链接库中方法my_dll_add加载失败"<< std::endl;FreeLibrary(hDll);return-1;}int a=2, b=3;int c=((PFN_ADD)pfn)(a, b);
	std::cout<<"my_dll_add("<< a<<","<< b<<") = "<< c<< std::endl;FreeLibrary(hDll);return0;}

函数指针类型的定义PFN_ADD,与函数“int my_dll_add(int a, int b)”的形式基本相同。

编译、链接、执行都可以,但是,就是代码“pfn = GetProcAddress(hDll, “my_dll_add”);”执行完毕后pfn的值是0,说明没有找到名为“my_dll_add”函数。

找不到库方法的解决办法

解决方法是在MyDll项目的“解决方案资源管理器”视图中,添加模块定义文件“Source.def”(右键单击源文件-添加-新建项-Visual C++代码-模块定义文件),

在这里插入图片描述
输入如下内容:

LIBRARY
EXPORTS
	my_dll_add	@1

如果是多个函数则

LIBRARY
EXPORTS
	my_dll_add	@1
	函数名1 @2
	函数名2 @2

重新生成项目MyDll,添加模块定义文件之后,生成项目的结果包含了lib文件。即生成结果包含了“E:\temp\MyDll\x64\Debug\MyDll.lib”和“E:\temp\MyDll\x64\Debug\MyDll.dll”。

并将新生成的“MyDll.dll”复制到“MyApp.exe”文件所在目录。

再次执行“MyApp.exe”,即正常找到库方法。运行结果如下:

在这里插入图片描述
动态库的使用-静态调用(使用头文件、lib文件、dll文件)
修改main函数,修改后“main.cpp”文件的内容如下:

#include <iostream>#include"E:\temp\MyDll\MyDllFunc.h"//	包含头文件intmain(){int a=2, b=3;int c=my_dll_add(a, b);
	std::cout<<"my_dll_add("<< a<<","<< b<<") = "<< c<< std::endl;return0;}

MyApp项目属性页,链接器,输入,附加依赖项,添加依赖项“E:\temp\MyDll\x64\Debug\MyDll.lib”
在这里插入图片描述
将动态库文件“E:\temp\MyDll\x64\Debug\MyDll.dll”复制到exe文件所在目录。可使用如下命令

          copy E:\temp\MyDll\x64\Debug\MyDll.dll E:\temp\MyApp\x64\Debug

完成上述操作,程序即可正常运行。

动态库静态调用-新增方法my_dll_sub-lib文件和库文件不一致会导致“找不到序数”的错误
在MyDll项目的头文件“MyDllFunc.h”中,添加方法声明:

intmy_dll_sub(int a,int b);

在“MyDllFunc.cpp”中,添加实现代码:

intmy_dll_sub(int a,int b){return a- b;}

在模块定义文件(.def文件)中添加

my_dll_sub	@2

生成解决方案。

在MyApp项目的main函数中添加调用函数my_dll_sub的代码

c=my_dll_sub(a, b);

在不更新“MyDll.dll”的情况下运行MyApp.exe,就会产生运行时错误,找不到序数:

无法定位序数2于动态链接库E:\temp\MyApp\x64\Debug\MyApp.exe上。
在这里插入图片描述
更新dll文件后,就没有上述运行时错误了。

这说明,找不到序数的错误,是由lib文件和dll文件不一致导致的。

导出DLL函数的3种方式
1. 源代码中的 __declspec(dllexport)

2. .def 文件中的 EXPORTS 语句

3. LINK 命令中的 /EXPORT 规范

 #pragma comment(linker,"/export:GetResult=_GetResult@8")

参考网页:导出DLL函数的3种方式_萧戈的专栏-CSDN博客

原文:https://blog.csdn.net/Alexabc3000/article/details/122990020

  • 作者:我不是萧海哇~~~~
  • 原文链接:https://wobushixiaohai.blog.csdn.net/article/details/123824982
    更新时间:2022-07-10 09:59:46