在windows平台和linux平台下都大量存在着库。本质上来说库是一种可执行的二进制代码(但不可以独立执行),可以被操作系统载入内存执行。
Linux下的库有两种:静态库和动态库(共享库)。
本文将介绍Linux下静态库和动态库的概念以及相应的创建与使用方法,文章内容为个人学习笔记,欢迎指正。
1 对比
| 类型 |
特点 |
| 静态库 |
在编译时被链接到程序中,作为可执行程序的一部分。在程序运行时不再依赖静态库,占用内存大 |
| 动态库 |
在可执行程序运行时载入内存。动态库已经在内存中不需要再次载入 |
2 静态库
2.1 概念
静态库指将所有相关的目标文件打包成为一个单独的文件,即静态库文件。
静态库以.a结尾,链接器会将程序中使用到的函数的代码从库文件中拷贝到程序中。
由于每个使用静态库的应用程序都需要拷贝所有函数的代码,所以静态链接的文件会比较大。
在Unix系统中,静态库以一种称为存档(archive)的特殊文件格式存放在磁盘中。
存档文件是一组连接起来的可重定位目标文件的集合,有一个头部用来描述每个成员目标文件的大小和位置。
2.2 静态库的创建和使用
2.2.1 环境准备
创建根文件夹staticDemo:
[root@VM_120_243_centos ~]
创建子文件夹include,用于存放所有头文件:
[root@VM_120_243_centos ~]
[root@VM_120_243_centos staticDemo]
进入include文件夹,编写myLib.h文件,申明printInfo()函数:
[root@VM_120_243_centos staticDemo]
[root@VM_120_243_centos include]
[root@VM_120_243_centos include]
void printInfo();
进入上层目录,创建lib文件夹,用于存放所有库文件,再其中编写print.c文件,使用了myLib.h中的printInfo()函数:
[root@VM_120_243_centos include]
[root@VM_120_243_centos staticDemo]
[root@VM_120_243_centos staticDemo]
[root@VM_120_243_centos lib]
[root@VM_120_243_centos lib]
void printInfo()
{
printf("print from print.c file...\n");
}
进入上层目录,编写main.c文件,用于测试:
[root@VM_120_243_centos lib]
[root@VM_120_243_centos staticDemo]
[root@VM_120_243_centos staticDemo]
int main(void)
{
printf("print from main.c file...\n");
printInfo();
return 0;
}
准备工作到此完成,整个目录结构如下:
[root@VM_120_243_centos staticDemo]# tree
.
├── include
│ └── myLib.h
├── lib
│ └── print.c
└── main.c
2 directories, 3 files
2.2.2 生成静态库
首先进入lib文件夹,生成print.o文件:
[root@VM_120_243_centos staticDemo]
[root@VM_120_243_centos lib]
[root@VM_120_243_centos lib]
print.c print.o
使用ar命令归档目标文件,得到静态库:
[root@VM_120_243_centos lib]
a - print.o
[root@VM_120_243_centos lib]
libprint.a print.c print.o
上述命令中crv是ar的命令选项:
-
c 如果需要生成新的库文件,不要警告
-
r 代替库中现有的文件或者插入新的文件
-
v 输出详细信息
使用-t参数可以查看静态库中包含的文件:
[root@VM_120_243_centos lib]
print.o
注意:我们要生成的库的文件名必须形如libxxx.a,这样我们在链接这个库时,就可以用-lxxx。
反过来讲,当我们告诉编译器-lxxx时,编译器就会在指定的目录中搜索libxxx.a或是libxxx.so。
2.2.3 生成可执行文件
进入上层目录,链接库并生成可执行文件:
[root@VM_120_243_centos staticDemo]
执行可执行文件:
[root@VM_120_243_centos staticDemo]# ./main
print from main.c file...
print from print.c file...
3 动态库
3.1 概念
动态库是一个目标模块,Linux系统以.so后缀表示,Windows以.dll后缀表示,使用动态库可以减小应用程序占用的空间和加载时间。
在运行时,可以加载到任意的存储器地址,并和一个再存储器中的程序链接起来,这个过程称为动态链接,是由一个叫做动态链接器的程序来执行的。
动态库是Linux系统最广泛的一种程序使用方式,工作原理是相同功能的代码可以被多个程序共同使用。
在程序加载的时候,内核会检查程序使用到的动态库是否已经加载到内存:
应用程序在自身加载时和运行过程中动态链接和加载动态库。
3.2 动态库的创建和使用
3.2.1 环境准备
创建根文件夹dynamicDemo:
[root@VM_120_243_centos ~]
创建子文件夹include,用于存放所有头文件:
[root@VM_120_243_centos ~]
[root@VM_120_243_centos dynamicDemo]
进入include文件夹,编写myLib.h文件,申明printInfo()函数:
[root@VM_120_243_centos dynamicDemo]
[root@VM_120_243_centos include]
[root@VM_120_243_centos include]
void printInfo();
进入上层目录,创建lib文件夹,用于存放所有库文件,再其中编写print.c文件,使用了myLib.h中的printInfo()函数:
[root@VM_120_243_centos include]
[root@VM_120_243_centos dynamicDemo]
[root@VM_120_243_centos dynamicDemo]
[root@VM_120_243_centos lib]
[root@VM_120_243_centos lib]
void printInfo()
{
printf("print from print.c file...\n");
}
进入上层目录,编写main.c文件,用于测试:
[root@VM_120_243_centos lib]
[root@VM_120_243_centos dynamicDemo]
[root@VM_120_243_centos dynamicDemo]
int main(void)
{
printf("print from main.c file...\n");
printInfo();
return 0;
}
准备工作到此完成,整个目录结构如下:
[root@VM_120_243_centos dynamicDemo]# tree
.
├── include
│ └── myLib.h
├── lib
│ └── print.c
└── main.c
2 directories, 3 files
3.2.2 生成动态库
进入lib文件夹,给gcc加入-fPIC参数,生成print.o文件:
[root@VM_120_243_centos dynamicDemo]
[root@VM_120_243_centos lib]
[root@VM_120_243_centos lib]
print.c print.o
使用gcc的-shared参数生成动态库libprint.so:
[root@VM_120_243_centos lib]
[root@VM_120_243_centos lib]
libprint.so print.c print.o
3.2.3 生成可执行文件
进入dynamicDemo目录,链接库并生成可执行文件:
[root@VM_120_243_centos dynamicDemo]
注意:如果同一目录下同时存在同名的动态库和静态库,比如libprint.so和libprint.a都在当前路径下,则gcc会优先链接动态库。
执行可执行文件:
[root@VM_120_243_centos dynamicDemo]# ./main
./main: error while loading shared libraries: libprint.so: cannot open shared object file: No such file or directory
系统报错找不到libprint.so,原来Linux是通过/etc/ld.so.cache文件搜寻要链接的动态库的。而/etc/ld.so.cache是ldconfig程序读取/etc/ld.so.conf文件生成的。(注意,/etc/ld.so.conf中并不必包含/lib和/usr/lib,ldconfig程序会自动搜索这两个目录)
这里我们指定环境变量来找到动态库路径,在下一节中会有详细介绍:
[root@VM_120_243_centos dynamicDemo]# LD_LIBRARY_PATH=./lib/ ./main
print from main.c file...
print from print.c file...
3.3 设置动态库的搜索路径
3.3.1 LD_LIBRARY_PATH
LD_LIBRARY_PATH是Linux环境变量名,该环境变量主要用于在查找动态库时搜索默认路径之外的其他路径。
通常,Linux搜索动态库的默认路径是/lib和/usr/lib。LD_LIBRARY_PATH路径优先级大于系统默认路径。
[root@VM_120_243_centos dynamicDemo]# gcc main.c -I./include/ -L./lib/ -lprint -o main
[root@VM_120_243_centos dynamicDemo]# LD_LIBRARY_PATH=./lib/ ./main
print from main.c file...
print from print.c file...
3.3.2 rpath
gcc的-R或者-rpath选项在编译链接时指定动态库路径,并且将动态库路径保存到可执行文件中。
运行可执行文件时直接到该路径下查找动态库,而不依赖于默认路径或者环境变量。
[root@VM_120_243_centos dynamicDemo]# gcc main.c -I./include/ -L./lib/ -lprint -Wl,-rpath=./lib -o main
[root@VM_120_243_centos dynamicDemo]# ./main
print from main.c file...
print from print.c file...
采用这种方式,每个程序可以设定独立的动态库位置,而不会像LD_LIBRARY_PATH环境变量那样影响其他程序。
3.3.3 ldconfig
ldconfig是一个动态链接库管理命令。该命令的用途是在默认路径(/lib和/usr/lib)以及动态库配置文件/etc/ld.so.conf内所包含的目录下,搜索动态库,然后为动态载入程序(ld.so)创建动态库列表的缓存文件(/etc/ld.so.cache)。
3.3.3.1 加入标准路径
这里以放入/usr.lib文件夹为例:
[root@VM_120_243_centos dynamicDemo]
更新缓存:
[root@VM_120_243_centos dynamicDemo]
查看动态库:
[root@VM_120_243_centos dynamicDemo]
libprint.so (libc6,x86-64) => /lib/libprint.so
编译执行:
[root@VM_120_243_centos dynamicDemo]# gcc main.c -I ./include/ -L ./lib/ -lprint -o main
[root@VM_120_243_centos dynamicDemo]# ./main
print from main.c file...
print from print.c file...
3.3.3.2 加入配置文件
首先查看/etc/ld.so.conf文件:
[root@VM_120_243_centos dynamicDemo]# cat /etc/ld.so.conf
include ld.so.conf.d
可以看到其包含了ld.so.conf.d目录,因此自定义配置文件libprint.conf加入到/etc/ld.so.conf.d文件夹中,配置文件中包含该程序的动态库路径:
[root@VM_120_243_centos dynamicDemo]
/root/dynamicDemo
[root@VM_120_243_centos dynamicDemo]
[root@VM_120_243_centos dynamicDemo]
/root/dynamicDemo/lib
更新缓存:
[root@VM_120_243_centos dynamicDemo]
查看动态库:
[root@VM_120_243_centos dynamicDemo]
libprint.so (libc6,x86-64) => /root/dynamicDemo/lib/libprint.so
编译执行:
[root@VM_120_243_centos dynamicDemo]# gcc main.c -I ./include/ -L ./lib/ -lprint -o main
[root@VM_120_243_centos dynamicDemo]# ./main
print from main.c file...
print from print.c file...
参考文献
[1] JollyWing. Linux静态库生成指南[EB/OL]. http://www.cnblogs.com/jiqingwu/p/4325382.html.
[2] JollyWing. Linux动态库生成与使用指南[EB/OL]. http://www.cnblogs.com/jiqingwu/p/linux_dynamic_lib_create.html.