静态库与动态库
1 静态库
1.1 静态库概述
静态库就是一些目标文件(一般以.o结尾)的集合,静态库一般以.a结尾,只用于生成可执行文件阶段。
链接步骤中,链接器从库文件中取所需代码,复制到生成的可执行文件中。
优点:可执行文件中包含了库代码的一份完整拷贝,在编译过程中被载入程序中。
缺点:多次使用会出现冗余拷贝,如果静态库有更新,所有使用它的程序都需要重新编译、发布。
1.2 静态库生成
#生成test.o目标文件
$ gcc -c test.c -o test.o
#使用ar命令将test.o打包成libtest.a静态库
# r:更新或增加新文件到静态库中
# c:创建一个库
# s:创建文档索引
$ ar rcs libtest.a test.o
#使用ar t 命令查看静态库内容
$ ar t libtest.a
test.o
2 动态库
2.1 动态库概述
动态库在链接阶段不会被复制到程序中,而是在程序运行时由系统动态加载到内存中供程序调用。
系统只需载入一次动态库,不同的程序可以得到内存中相同动态库的副本,因此节省内存。
2.2 动态库生成
#生成目标文件
$ gcc -c test.c
#使用-fPIC和-shared生成动态库
#-fPIC:创建与地址无关的代码
$ gcc -shared -fPIC -o libtest.so test.o
3 静态库与动态库的使用
3.1 准备文件
准备三个文件:
// tool.h
int find_max(int arr[], int n);
// tool.c
#include "tool.h"
int find_max(int arr[], int n){
int max = arr[0];
int i;
for(i = 0; i < n; i++){
if(arr[i] > max){
max = arr[i];
}
}
return max;
}
// main.c
#include <stdio.h>
#include "tool.h"
int main(){
int arr[] = {1,3,5,8,2};
int max = find_max(arr,5);
printf("max = %d\n", max);
return 0;
}
3.2 静态库的使用
# 生成目标文件
$ gcc -c tool.c
# 生成静态库
$ ar rcs libtool.a tool.o
# -l:指定要链接的库,库名省略前缀lib
# -L:编译程序按照-L指定的目录查找库文件;“.”表示当前目录
$ gcc -o main main.c -L. -ltool
$ ./main
max = 8
# 查看可执行文件依赖了哪些库
$ ldd main
linux-vdso.so.1 (0x00007fff4d1b9000)
libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007efc9007f000)
/lib64/ld-linux-x86-64.so.2 (0x00007efc90672000)
3.3 动态库的使用
# 生成目标文件
$ gcc -c tool.c
# 生成静态库
$ gcc -shared -fPIC -o libtool.so tool.o
$ gcc -o main main.c -L. -ltool
# 将当前目录设置到环境变量中,否则找不到libtool.so库
$ LD_LIBRARY_PATH=/home/louis/root/test
$ ./main
max = 8
4 静态库与动态库的区别
静态库在程序编译时会连接到目标代码中,程序运行时不再需要静态库,因此体积较大。而且每次编译都需要载入静态代码,因此内存开销大。
动态库在程序编译时不会被链接到目标代码中,而是在程序运行时才被载入,程序运行时需要动态库存在,因此体积较小。而且系统只需载入一次动态库,不同程序可以得到内存中相同的动态库副本,因此内存开销小。