一般来说项目偏爱生成dll动态库文件,因为可以解决静态库造成的空间浪费和更新困难问题,另外创建静态库时,我一般是建立空项目后,在项目配置类型中进行选择。
什么是库
库是写好的现有的,成熟的,可以复用的代码。现实中每个程序都要依赖很多基础的底层库,不可能每个人的代码都从零开始,因此库的存在意义非同寻常。本质上来说库是一种可执行代码的二进制形式,可以被操作系统载入内存执行。库有两种:静态库(.a、.lib)和动态库(.so、.dll)。
所谓静态、动态是指链接。回顾一下,将一个程序编译成可执行程序的步骤:
静态库
之所以称为【静态库】,是因为在链接阶段,会将汇编生成的目标文件.o与引用到的库一起链接打包到可执行文件中。因此对应的链接方式称为静态链接。
试想一下,静态库与汇编生成的目标文件一起链接为可执行文件,那么静态库必定跟.o文件格式相似。其实一个静态库可以简单看成是一组目标文件(.o/.obj文件)的集合,即很多目标文件经过压缩打包后形成的一个文件。
静态库优缺点总结:
1. 静态库对函数库的链接是放在编译时期完成的。
2. 程序在运行时与函数库再无瓜葛,移植方便。
3. 浪费空间和资源,因为所有相关的目标文件与牵涉到的函数库被链接合成一个可执行文件。
Windows下创建与使用静态库
创建静态库(.lib)
创建静态库有两种方法。
(1)创建项目时进行设置
创建win32控制台程序时,勾选静态库类型(下面的预编译等选项可选可不选);打开工程“属性面板”-”配置属性”-”常规”,配置类型选择静态库。
(2)生成项目时进行设置
创建空项目之后,生成项目时,选择项目-属性-配置属性-常规-项目默认值-配置类型-选择生成静态库lib。
程序编译通过后,Build项目即可生成静态库。即可生成.lib,输出栏会显示出静态库的位置。
拷贝到工程目录下
右击项目 - 在文件资源管理器中打开文件夹 - 将文件拷贝至工程目录下
使用静态库
如果将静态链接库和头文件拷贝到工程目录下,则可以只进行这一步:“属性面板”-”配置属性”-“链接器”-”输入”,附加依赖库中输入静态库名StaticLibrary.lib,或者采用以下语句:
1 #pragma comment(lib, "./StaticLibrary.lib");
如果没有进行拷贝,则需要通过“属性面板”-”配置属性”-“VC++目录”-”库目录”,将路径添加到工程中,另外在附加依赖库目录中输入静态库所在目录。
动态库
通过上面的介绍发现静态库,容易使用和理解,也达到了代码复用的目的,那为什么还需要动态库呢?
为什么还需要动态库(静态库的缺点)?
为什么需要动态库,其实也是静态库的特点导致。
1.空间浪费是静态库的一个问题。
2.另一个问题是静态库对程序的更新、部署和发布也会带来麻烦。如果静态库lib更新了,所有使用它的应用程序都需要重新编译、发布给用户(对于玩家来说,可能是一个很小的改动,却导致整个程序重新下载,全量更新)。
动态库在程序编译时并不会被连接到目标代码中,而是在程序运行时才被载入。不同的应用程序如果调用相同的库,那么在内存里只需要有一份该共享库的实例,规避了空间浪费问题。动态库在程序运行时才被载入,也解决了静态库对程序的更新、部署和发布页会带来麻烦。用户只需要更新动态库即可,增量更新。
动态库特点总结:
1.动态库把对一些库函数的链接载入推迟到程序运行的时期。
2.可以实现进程之间的资源共享。(因此动态库也称为共享库)
3.将一些程序升级变得简单。
4.甚至可以真正做到链接载入完全由程序员在程序代码中控制(显示调用)。
Windows下创建与使用动态库
创建动态库(.dll)
创建动态库同样有两种方法,创建前设置和生成项目时设置。
(1)创建前设置
创建前设置步骤如下:单击“新建项目”,选择“Win32 项目”,选择“下一步”,然后到了如下界面:点击“DLL”,并同时点击“导出符号”。然后点击“完成”。“导出符号”,将为我们创建生成DLL的模板。仔细观察,vs自动为我们创建了示例,导出变量nMakeDLL,导出函数fnMakeDLL和导出类CMakeDLL。如果上面没有点击“导出符号”,将不会生成这些。所以为了方便DLL的创建,我们最好还是使用“导出符号”。
先看下面这段代码:
1 #ifdef MAKEDLL_EXPORTS 2 3 #define MAKEDLL_API __declspec(dllexport) 4 5 #else 6 7 #define MAKEDLL_API __declspec(dllimport) 8 9 #endif
如果定义了MAKEDLL_EXPORTS,然后我们再定义MAKEDLL_API为__declspec(dllexport),否则我们定义__declspec(dllimport)。前者主要为生成DLL的项目所使用,后者为使用DLL的项目所使用。dllexport和dllimport存储类特性是C和C++语言的Microsoft专用扩展。可以使用它们从 DLL 中导出或向其中导入函数、数据和对象。
为什么要使用__declspec(dllexport)与__declspec(dllimport),及两者的区别:都是DLL内的关键字,即导出与导入。他们是将DLL内部的类与函数以及数据导出与导入时使用的。主要区别在于,dllexport是在这些类、函数以 及数据的申明的时候使用。用过表明这些东西可以被外部函数使用,即(dllexport)是把DLL中的相关代码(类,函数,数据)暴露出来为其他应用程 序使用。使用了(dllexport)关键字,相当于声明了紧接在(dllexport)关键字后面的相关内容是可以为其他程序使用的。而 dllimport关键字是在外部程序需要使用DLL内相关内容时使用的关键字。当一个外部程序要使用DLL内部代码(类,函数,全局变量)时,只需要在 程序内部使用(dllimport)关键字声明需要使用的代码就可以了,即(dllimport)关键字是在外部程序需要使用DLL内部相关内容的时候才使用。(dllimport)作用是把DLL中的相关代码插入到应用程序中。如果没有使用dllexport导出函数,将无法生成lib文件,只能生成dll文件。
接下来,你可以改写上面的类和函数,然后点击项目,选择“生成”即可产生DLL和LIB。为了简化起见,本例只考虑导出函数。
修改完成后点击Build项目,在输出栏显示dll文件的生成位置。
(2)生成项目时设置
这种方法要求生成动态库需要设置工程属性,打开工程“属性面板”-”配置属性”-”常规”,配置类型选择动态库。但这种方法是建立空项目,因此,需要在头文件中手动设置__declspec(dllexport)等语句。
可以直接使用上面那段代码,也可以只在导出的函数前加__declspec(dllexport),注意:前面是两条下划线
生成的DLL文件如下图所示:
这里可能大家有个疑问,动态库怎么还有一个DynamicLibrary.lib文件?即无论是静态链接库还是动态链接库,最后都有lib文件,那么两者区别是什么呢?其实,两个是完全不一样的东西。
StaticLibrary.lib的大小为190KB,DynamicLibrary.lib的大小为3KB,静态库对应的lib文件叫静态库,动态库对应的lib文件叫导入库。实际上静态库本身就包含了实际执行代码、符号表等等,而对于导入库而言,其实际的执行代码位于动态库中,导入库只包含了地址符号表等,确保程序找到对应函数的一些基本地址信息。
使用动态库
在引用一个新的dll时,我们一般需要dll的.h,.lib和.dll三个文件。
将三个文件拷贝到工程目录下,对于lib和头文件的使用方法与静态链接库相同,这时直接运行程序,可能会出现一个错误,说“缺少什么dll”,我们只需将此前生成的dll,放在debug或release目录下,即可编译通过。
使用方法总结
.h,.lib,.dll的添加方法其实各有很多种。
.h有两种:
1. 头文件较少时,直接放在工程目录下,#include "*.h"
2. 建立include文件夹,放在里面,然后在项目属性的“VC++目录”-》“包含目录”,添加.h路径即可。
.lib的方法有两种
1.lib文件较少时,直接放在工程目录下,#pragma comment(lib, "testCpp.lib")
2.建立lib文件夹,目属“VC++目录”-》“库目录”添加路径,然后在项目属性的“链接器”-》“输入”-》“附加依赖项”,添加.lib名字。
dll的方法有两种
1.直接放在debug/release目录下
2.建立bin文件夹,放在里面,然后在项目“环境”中添加bin的路径(PS:最好不要用这个)
以上皆可以自由组合。
-------------------------------------------------------------------------------------------------
如果上面的资料对你有启发,麻烦点个推荐,让更多人的人看到哦。
关注公众号【两猿社】,懂点互联网,懂点IC的程序猿,带你丰富项目经验哦