库文件可以理解为别人写好的现成的代码,但是看不见源码,只提供程序入口。库又分为动态库和静态库,静态库是在编译的时候将库编译进可执行程序中,运行时不再依赖库文件,而动态库是在运行时加载,运行时需要依赖库文件。静态库以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命令和它的参数,在这里不作具体分析。