库文件可以理解为别人写好的现成的代码,但是看不见源码,只提供程序入口。库又分为动态库和静态库,静态库是在编译的时候将库编译进可执行程序中,运行时不再依赖库文件,而动态库是在运行时加载,运行时需要依赖库文件。静态库以libxxx.a命名,动态库以libxxx.so命名。
比如编写两个文件,hello.c和main.c,源码如下:
hello.c
1 #include <stdio.h>
2
3 void hello(void)
4 {
5 printf("Hello World\n");
6 }
main.c
1 extern void hello(void);
2
3 int main(int argc, const char *argv[])
4 {
5 hello();
6
7 return 0;
8 }
可以用gcc hello.c main.c 命令将两个文件编译成一个可执行文件,但是如果我不希望hello.c中的源代码被人看到,我就会将hello.c制作成库文件。
生成静态库
执行命令:gcc -c hello.c -o hello.o 得到hello.o文件
执行命令:ar crs libhello.a hello.o 得到libhello.a文件,ar命令的具体解释可以参照这篇博客:https://blog.csdn.net/hxg130435477/article/details/8217247
这时就可以使用libhello.a文件了,执行命令:gcc main.c -L. -lhello -o hello
得到可执行文件hello。
需要注意的几个地方:
1、使用ar命令生成的静态库需要由.o文件生成,如果使用命令ar crs libhello.a hello.c 也能生成名为libhello.a的文件,但生成的libhello.a不能用,编译时会报错:
“could not read symbols: Archive has no index; run ranlib to add one collect2: ld returned 1 exit status”
2、生成的.a文件一定要以lib开头,比如libhello.a,而在指定库名时却不能加上最前面的“lib”,否则编译时会都提示找不到库。
ar crs libhello.a hello.o //正确
ar crs hello.a hello.o //错误
gcc main.c -L. -lhello //正确
gcc main.c -L. -llibhello //错误
3、-L用于指定库的路径,与路径之间可以用空格隔开,也可以不用空格。-l用于指定库名,与库名之间可以用空格隔开,也可以不用
gcc main.c -L. -lhello //正确
gcc main.c -L . -l hello //正确
4、如果使用-L指定的路径中没有-l指定的库,或者编译时不加-L参数,那么编译器会去系统默认的库路径中寻找,默认路径为/lib。测试的时候将libhello.a拷贝到/usr/lib或者/lib目录下,测试完之后删除该文件需要使用:sudo rm -f
同时编译多个文件
假设存在3个文件,hello.c,how.c和main.c,在当前路径下生成了hello.o,libhello.a,how.o,libhow.a,只编译c文件可以用:
gcc main.c hello.c how.c
how.c 和libhello.a
gcc main.c how.c -L . -lhello
libhow.a和libhello.a
gcc main.c -L . -lhello -lhow
上面这种情况由于libhello.a和libhow.a在同一路径,所以只需要使用-L指定一次路径,而如果这两个文件不在同一个路径,那么就需要分别用-L指定路径,比如libhello.a在./dir目录下时:
gcc main.c -L . -lhow -L ./dir -lhello
多个文件合并
1、将多个.o文合并为一个.a文件:
ar crs libnew.a hello.o how.o
gcc main.c -L . -lnew
2、将.o文件加入.a文件:
ar crs libhello.a how.o
3、将多个.a文件合并为一个新的.a文件
ar crs libhello.a hello.o
ar crs libhow.a how.o
ar crsT libnew.a libhello.a libhow.a
gcc main.c -L. -lnew
注意执行前要先删掉所有的.a文件,第三步中要加参数"T",否则生成的libnew.a不能使用,编译不过,如果说第3步没有加“T”参数,在不删除.a的情况下重复执行123步,得到的libnew.a也不能使用。原因需要看ar命令和它的参数,在这里不作具体分析。