Linux 库的制作
1.动态库 xxx.so
2.静态库 xxx.a
相同点:
库中都有一组实现好的函数接口
不同点:
如果我们连接的是动态库,此时只是在可执行文件的头信息中记录动态库的名字,具体函数的实现代码并没有存放在
可执行文件中,所以在可执行文件执行的时候,需要将它依赖的动态库载入内存
可执行文件头信息的查看:
readelf -a 可执行文件名
可执行文件依赖的动态库
readelf -a 可执行文件名 | grep "Shared"
|elf头 |
|--------|
|机器码|
| |
如果我们连接的是静态库,此时会将库函数中,函数具体实现代码,拷贝到我们的可执行文件中。此时
可执行文件执行的时候,不需要静态库的支持。
3.制作静态库
(1)将 xx.c编译成 xxx.o
gcc -c xx.c -o xxx.o
(2)将所有的 xxx.o文件打包成一个静态库
ar -cr lib库名.a *.o
查看库中的函数信息:
nm xxx.a
4.gcc 一些编译参数
-I 指定头文件搜索路径
-L 指定需要连接库的路径
-l 指定需要连接库的名字
例如:
gcc main.c -I 指定头文件搜索路径 -L 指定连接库所在路径 -l指定连接库的名字
注意:
(1)gcc 默认搜索头文件路径 /usr/include
(2)gcc 默认搜索库文件路径 /lib 和 /usr/lib
(3)gcc 默认只能识别它自带库,而第三方库无法直接识别
4.动态库制作
(1)将 xx.c编译成 xxx.o
gcc -c xx.c -o xxx.o
(2)将所有的 xxx.o 打包成一个动态库
gcc -shared *.o -o lib库名.so
程序运行出现的错误:
./a.out: error while loading shared libraries: libadd_test.so: cannot open
shared object file: No such file or directory
原因:
因为连接的是动态库,所以在可执行文件执行的时候,操作系统需要将他依赖的动态库载入内存运行
,但是操作系统默认加载库的路径:
<1>环境变量 LD_LIBRARY_PATH 记录的路径中寻找
<2>/lib 或 /usr/lib
注意:
如果动态库和静态库同时存在,gcc默认连接的是动态库,如果强制性连接静态库需要加-static参数
5.实例展示如下:
test.c:
add.c:
现在开始分别生成动态库和静态库:
静态库:
gcc -c test.c -o test.o
gcc -c add.c -o add.o
ar -cr libadd_test.a *.o
然后查看nm libadd_test.a
动态库
gcc -c test.c -o test.o
gcc -c add.c -o add.o
gcc -shared *.o -o libadd_test.so
分别测试动态库和静态库
test.h
main.c(需要#include <test.h>)
gcc main.c -I . -L . -ladd_test
如果是静态库,可以直接执行
如果是动态库,需要
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:.
然后再执行.
具体如果需要查看可执行文件需要的动态库,可以执行如下命令:
readelf -a ./a.out |grep "Shared"