C/C++的动态库和静态库
静态库
在链接阶段,将汇编生成的目标文件.o与引用到的库一起链接打包到可执行文件(.a)就是静态库。
静态库特点
- 静态库对函数库的链接是放在编译时期完成的。
- 程序在运行时与函数库再无瓜葛,移植方便。
- 浪费空间和资源,因为所有相关的目标文件与牵涉到的函数库被链接合成一个可执行文件。
生成静态库
假设有三个源文件helloworld_1.c helloworld_2.c helloworld.h
$ gcc -c helloworld_1.c helloworld_2.c #生成helloworld_1.o helloworld_2.o
$ ar crv libhelloworld.a helloworld_1.o helloworld_2.o
ar
命令用来生成静态库 ,crv
是命令选项:
- c 如果需要生成新的库文件,不要警告
- r 代替库中现有的文件或者插入新的文件,一般必选
- v 输出详细信息
注意:生成的库名必须遵守规范,lib[static_library_name].a
:lib为前缀,中间是静态库名,扩展名为.a。
此外,静态库提供者同时应该提供头文件,使用该静态库的程序,必须包含头文件。
静态库的两大缺点:
- 空间浪费:静态库参与形成的每个可执行程序将静态库拷贝,并加载进内存,即每个用到该静态库的进程会单独拥有一份该静态库的拷贝。同时,静态库生成的时候,会将其调用的所有的库打包在一起,而动态库却是只将自身的代码生成库文件。因此,相同代码生成的静态库本来就比动态库大,再加上调用时被拷贝多份,自然会浪费很多内存空间。
- 另一个问题是静态库对程序的更新、部署和发布页会带来麻烦。如果静态库libhelloworld.a更新了,所以使用它的应用程序都需要重新编译、发布给用户(对于玩家来说,可能是一个很小的改动,却导致整个程序重新下载,全量更新)。
动态库
动态库在程序编译时并不会被连接到目标代码中,而是在程序运行是才被载入。不同的应用程序如果调用相同的库,那么在内存里只需要有一份该共享库的实例,规避了空间浪费问题。
动态库在程序运行是才被载入,也解决了静态库对程序的更新、部署和发布页会带来麻烦。用户只需要更新动态库即可,增量更新。
动态库特点
- 动态库把对一些库函数的链接载入推迟到程序运行的时期。
- 可以实现进程之间的资源共享。(因此动态库也称为共享库)
- 将一些程序升级变得简单。
- 甚至可以真正做到链接载入完全由程序员在程序代码中控制(显式调用)。
生成动态库
假设有三个源文件helloworld_1.c helloworld_2.c helloworld.h
以下