动态链接库的生成和使用,从入门到精通
1. 动态链接库和静态链接库的区别※
静态链接库(.LIB) 由函数和数据编译而成的一个二进制文件。使用时,在编译链接阶段,由链接器从库中复制这些函数和数据,并把他们与应用程序的其他模块组合起来创建最终的可执行文件。由于静态链接库中的程序和数据已经被复制并应用到可执行文件中,因此发布产品时不需要发布使用的静态库文件。
动态链接库 (.DLL) 包含被可执行程序或其他DLL调用来完成某项工作的函数,不可以直接运行,也不可以接收消息。动态链接库一半包含两个文件,引入库文件(.LIB)和动态链接库文件(.DLL)。使用时,在编译链接阶段,只需要链接引入库文件,动态链接库中的函数和数据并不复制到程序中,在运行阶段去访问DLL文件中的函数。由于动态链接库中函数和数据并没有被复制,因此发布产品时,必须包含动态链接库文件。
- 引入库文件 (.LIB): 包含该动态链接库包含的函数和变量的符号名。注意:虽然引入库文件和静态链接库文件的后缀名相同(.LIB),但是他们之间有着本质的区别,不可混淆。
- 动态链接库文件 (.DLL): 包含该动态链接库实际的函数和数据。在程序运行阶段,加载该文件,并将该文件映射到进程地址空间中,然后访问该文件中的相应函数。
2. 创建和使用动态链接库(VS2019)※
方法一:※
创建动态链接库:※
- 第一步: 创建新工程,修改项目属性为“动态库”,配置属性–>常规–>配置类型–>动态库;
- 第二步: 新建源文件,添加要导出到动态库中的函数,与常规函数的区别是在函数前面添加标识符“_declspec(dllexport)”;如:
_declspec(dllexport) int myadd(int a, int b)
{
return a + b;
}
- 第三步: 生成动态链接库:生成–>生成【项目名称】;
隐式使用:※
- 第一步: 添加引入库文件(.LIB),即,将引入库文件拷贝到当前程序目录或设置附加库目录,并设置附加依赖项;
- 第二步: 引入动态链接库文件,即,将动态链接库文件(.DLL)放入执行程序可搜索到的任意目录中 1;
- 第三步: 声明外部函数,即,在使用外部函数之前,使用extern或_declspec(dllimport)声明外部函数,如:
extern int myadd(int a, int b);
extern int mysubtract(int a, int b);
或
_declspec(dllimport) int myadd(int a, int b);
_declspec(dllimport) int mysubtract(int a, int b);
- 第四步: 按普通函数的调用方法调用动态链接库的文件即可,如:
int result = myadd(2,3);
方法一使用心得: 使用该方法创建的动态链接库需要声明外部函数,即需要用户知道该动态链接库中函数的具体名称,声明的外部函数需要与动态链接库中的函数名称及参数类型完全一致,否则就会报错。因此使用这种方法创建的动态链接库十分不方便,而且非常容易出错。为了解决这个问题,对方法一进行了升级。
方法二:※
创建动态链接库:※
- 第一步: 创建新工程,修改项目属性为“动态库”,配置属性–>常规–>配置类型–>动态库;
- 第二步: 新建源文件,添加要导出到动态库中的函数,与常规函数的区别是在函数前面添加标识符“_declspec(dllexport)”;
- 第三步: 在头文件添加函数的外部调用声明供使用该动态链接库的客户使用 ,如,在Calculation.h中添加下列声明:
_declspec(dllimport) int myadd(int a, int b);
_declspec(dllimport) int mysubtract(int a, int b);
- 第四步: 生成动态链接库:生成–>生成【项目名称】;
隐式使用:※
- 第一步: 添加引入库文件,即,将引入库文件拷贝到当前程序目录或设置附加库目录,并设置附加依赖项;
- 第二步: 引入动态链接库文件,即,将动态链接库文件(.DLL)放入执行程序可搜索到的任意目录中 1;
- 第三步: 包含动态链接库的头文件,即将该头文件拷贝到工程目录下,或设置项目属性中的包含路径,并#include<xxxx.h>;
- 第四步: 按普通函数的调用方法调用动态链接库的文件即可;
方法二使用心得: 该方法在创建动态链接库时就提供了包含所有函数外部调用原型的头文件,用户使用时,只需包含该头文件即可,不再需要自己定义外部调用声明,提高了效率,降低了调用出错率。但是,此时创建的头文件,并没有被该动态链接库自身使用,所以,即使我们删除掉该头文件,依然可以生成动态链接库,该头文件仅仅是为动态链接库的使用提供了便利。使用这种方法,如果我们编写头文件时不小心出错(例如,拼写错误,参数类型错误,参数数量错误等),我们使用动态链接库时就会出错。
方法三:※
创建动态链接库:※
- 第一步: 创建新工程,修改项目属性为“动态库”,配置属性–>常规–>配置类型–>动态库;
- 第二步: 在头文件添加外部函数导入声明,供使用该动态链接库的客户和该动态链接库自身使用,即,利用条件编译指令判断是客户使用还是库自身使用,从而利用不同定义的同名宏代替函数前面的标识符;
#pragma once
#ifdef MY_CAL_API
#else
#define MY_CAL_API _declspec(dllimport)
#endif // MY_CAL_API
MY_CAL_API int myadd(int a, int b);
- 第三步: 在源文件中添加宏定义,并包含具有函数声明的头文件;
#define MY_CAL_API _declspec(dllexport)
#include "Calculation.h"
- 第四步: 在源文件中添加要导出到动态库中的函数,函数前面不再需要标识符“_declspec(dllexport)”;
int myadd(int a, int b)
{
return a + b;
}
- 第五步: 生成动态链接库:生成–>生成【项目名称】;
隐式使用:※
与方法二中的使用方法相同。
方法三使用心得: 该方法在创建动态链接库时提供的包含所有函数外部调用原型的头文件不仅为用户提供了方便,还为动态链接库中的函数提供了原型。该方法保证了提供的头文件中的函数原型与动态链接库中的函数完全一致,增加了程序的稳健性。而且,函数定义前不再需要任何标识符,降低了动态库编写的工作量。
3. 导出类到动态链接库※
- 导出整个类: 只需要在Class关键字和类名之间加入导出标识符即可,函数声明之前不需要任何处理。
- 导出类中的某个成员函数: 在该成员函数声明之前加入导出标识符即可,Class关键字和类名之间不再需要导出标识符。使用该函数时,仍然使用成员函数的调用方法。
4. 显式加载动态链接库※
上面是常用的动态链接库的使用方法——隐式加载,下面介绍一种不常用的动态链接库的使用方法——显式加载,即,使用函数加载动态链接库,使用函数加载动态链接库中的函数。具体的流程如下:
- 第一步: 引入动态链接库文件,即,将动态链接库文件(.DLL)放入执行程序可搜索到的任意目录中 1;
- 第二步: 使用LoadLibrary函数将指定的动态链接库映射到调用进程的地址空间,并返回该动态链接库的句柄,该函数的使用方法自行查阅。
HMODULE LoadLibrary(LPCTSTR lpFileName);
- 第三步: 使用GetProcAddress函数获取要调用的函数的地址,并返回该函数的指针,该函数的使用方法请自行查阅,函数原型如下:
FARPROC GetProcAddress(HMODULE hModule, LPCSTR lpProcName);
- 第四步: 应用函数指针完成函数调用功能;
- 第五步: 在完成调用之后,使用FreeLibrary将动态链接库从地址空间中卸载,函数原型如下:
BOOL FreeLibrary(HMODULE hModule);
执行程序可搜索的目录依次为:程序的执行目录(Debug目录) --> 当前目录(一般为头文件和源文件所在的目录) --> 系统目录(C:\WINNT\system32,C:\WINNT\system,C:\WINNT)–>path 环境中列出的路径 ↩︎ ↩︎ ↩︎