使用VS2022创建动态库时,请使用模块定义文件(.def文件)创建项目时,仅生成dll文件,发现没有lib文件的有趣现象(或奇怪现象)。模块定义文件(.def文件)时,dll和lib文件将包含在结果中。
动态库的创建
使用VS2022新建动态链接库项目,项目名为MyDll,项目所在文件夹为“E:\temp”。
点击创建,即完成项目的创建。
在VS2022的“解决方案资源管理器”视图中,添加名为“MyDllFunc.h”(文件名可以任意)的头文件(右键单击“头文件-添加-新建项-头文件”),输入代码如下:
#ifndef _MY_DLL_FUNC_H_
#define _MY_DLL_FUNC_H_
int my_dll_add(int a, int b);
#endif
在VS2022的“解决方案资源管理器”视图中,添加名为“MyDllFunc.cpp”(文件名可以任意)的C++文件(右键单击“源文件-添加-新建项-C++文件”),输入如下代码:
#include ";
#include "MyDllFunc.h"
int my_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>
int main()
{
return 0;
}
生成解决方案,会得到可执行文件“E:\temp\MyApp\x64\Debug\MyA”,将动态库文件“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>
int main()
{
HINSTANCE hDll = 0;
hDll = LoadLibrary("MyDll.dll");
if (!hDll) {
std::cout << "动态链接库 MyDll.dll 加载失败" << std::endl;
return -1;
}
typedef int (*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);
return 0;
}
函数指针类型的定义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
重新生成项目MyDll,添加模块定义文件之后,生成项目的结果包含了lib文件。即生成结果包含了“E:\temp\MyDll\x64\Debug\MyDll.lib”和“E:\temp\MyDll\x64\Debug\MyDll.dll”。
并将新生成的“MyDll.dll”复制到“MyA”文件所在目录。
再次执行“MyA”,即正常找到库方法。运行结果如下:
动态库的使用-静态调用(使用头文件、lib文件、dll文件)
修改main函数,修改后“main.cpp”文件的内容如下:
#include <iostream>
#include "E:\temp\MyDll\MyDllFunc.h" // 包含头文件
int main()
{
int a = 2, b = 3;
int c = my_dll_add(a, b);
std::cout << "my_dll_add(" << a << "," << b << ") = " << c << std::endl;
return 0;
}
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”中,添加方法声明:
int my_dll_sub(int a, int b);
在“MyDllFunc.cpp”中,添加实现代码:
int my_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”的情况下运行MyA,就会产生运行时错误,找不到序数:
无法定位序数2于动态链接库E:\temp\MyApp\x64\Debug\MyA上。
更新dll文件后,就没有上述运行时错误了。
这说明,找不到序数的错误,是由lib文件和dll文件不一致导致的。
导出DLL函数的3种方式
1. 源代码中的 __declspec(dllexport)
2. .def 文件中的 EXPORTS 语句
3. LINK 命令中的 /EXPORT 规范 #pragma comment(linker,"/export:GetResult=_GetResult@8")
参考网页:
1.《网页上的动态怎么设计软件?终于找到答案了动态库的创建和使用(Win10+VS2022)》援引自互联网,旨在传递更多网络信息知识,仅代表作者本人观点,与本网站无关,侵删请联系页脚下方联系方式。
2.《网页上的动态怎么设计软件?终于找到答案了动态库的创建和使用(Win10+VS2022)》仅供读者参考,本网站未对该内容进行证实,对其原创性、真实性、完整性、及时性不作任何保证。
3.文章转载时请保留本站内容来源地址,https://www.lu-xu.com/gl/3229038.html