1 静态库的命名
静态库可以认为是一些目标代码的集合,是在可执行程序运行前就已经加入到执行码中,成为执行程序的一部分。 按照习惯,一般以“.a”做为文件后缀名。静态库的命名一般分为三个部分:
- 前缀:lib
- 库名称:自己定义即可
- 后缀:.a
所以最终的静态库的名字应该为:libxxx.a
2 静态库制作
demo代码
// add.h
int add(int a, int b);
// add.c
int add(int a, int b)
{
return a + b;
}
// sub.h
int sub(int a, int b);
// sub.c
int sub(int a, int b);
// main.c
#include <stdio.h>
#include "add.h"
#include "sub.h"
int main()
{
int a = add(10, 20);
int b = sub(10, 20);
printf("add(10,20) = %d\n",a);
printf("sub(10,20) = %d\n",b);
}
步骤1:将c源文件生成对应的.o文件
gcc -c add.c -o add.o
gcc -c sub.c -o sub.o
步骤2:使用打包工具ar将准备好的.o文件打包为.a文件 libtest.a
ar -rcs libtest.a add.o sub.o
在使用ar工具是时候需要添加参数:rcs
- r 更新
- c 创建
- s 建立索引
3 静态库使用
静态库制作完成之后,需要将.a文件和头文件一起发布给用户。测试文件为main.c,静态库文件为libtest.a
gcc main.c -L ./ -I ./ -ltest -o test
参数说明:
- -L:表示要连接的静态库所在目录
- -I:指定链接时头文件的路径(大写的i)
- -l :指定链接时需要的库,去掉前缀和后缀(小写的 L)
编译生成可执行程序test后执行
4 动态库的命名
共享库在程序编译时并不会被连接到目标代码中,而是在程序运行是才被载入。不同的应用程序如果调用相同的库,那么在内存里只需要有一份该共享库的实例,规避了空间浪费问题。动态库在程序运行是才被载入,也解决了静态库对程序的更新、部署和发布页会带来麻烦。用户只需要更新动态库即可,增量更新。
按照习惯,一般以“.so”做为文件后缀名。共享库的命名一般分为三个部分:
- 前缀:lib
- 库名称:自己定义即可
- 后缀:.so
所以最终的动态库的名字应该为:libxxx.so
5 动态库的制作
步骤一:生成目标文件,此时要加编译选项:-fPIC(fpic)
gcc -fpic -c add.c -o add. o
gcc -fpic -c sub.c -o sub.o
参数:-fPIC 创建与地址无关的编译程序(pic,position independent code),是为了能够在多个应用程序间共享。
步骤二:生成共享库,此时要加链接器选项: -shared(指定生成动态链接库)
gcc -shared add.o sub.o -o libtest.so
6 动态库的使用
1 引用动态库编译成可执行文件(跟静态库方式一样)
gcc main.c -L ./ -I ./ -ltest -o test
编译生成可执行程序test后执行,但是程序报错
./test: error while loading shared libraries: libtest.so: cannot open shared object file: No such file or directory
编译时虽然告诉了动态库libtest.so的路径,但是生成程序文件的时候动态库并没有加载到可执行程序里面,所以要想程序正确的运行,需要在程序运行的时候告诉所依赖的动态库的路径
2 让系统找到动态库(下面例举了5种方法)
1、拷贝自己制作的共享库到/lib或者/usr/lib
2、临时设置LD_LIBRARY_PATH:
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:库路径
3、永久设置 LD_LIBRARY_PATH,
把export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:库路径,设置到~/.bashrc或者 /etc/profile文件中
4、 将动态库的路径添加到 /etc/ld.so.conf文件中,编辑/etc/ld.so.conf文件,加入库文件所在目录的路径,运行sudo ldconfig -v,该命令会重建/etc/ld.so.cache文件
5、使用符号链接, 但是一定要使用绝对路径
ln -s /home/test/libtest.so /lib/libtest.so