使用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