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

最新文章

  1. VC中对文件的读写
  2. codeforces 505A. Mr. Kitayuta&#39;s Gift 解题报告
  3. php-eclipse乱码处理
  4. Android Studio 系列教程(转载)
  5. 看到的一些js小知识
  6. “String.h” 源代码总结
  7. Spring - Bean的概念及其基础配置
  8. JS判断浏览器类型与版本
  9. intelliJ IDEA自动优化导入包设置
  10. [LeetCode] Range Addition II 范围相加之二
  11. SpringMVC表当重复提交
  12. 8.0.11版本的mysql更改root密码
  13. OpenCV3 for python3 学习笔记1
  14. Maven的课堂笔记1
  15. 利用node,跑项目。
  16. python实现Excel删除特定行、拷贝指定行操作
  17. OPNET仿真软件资料合集
  18. Day Ten
  19. [BZOJ5351]Query on a sequence
  20. 亚马逊EC2

热门文章

  1. CTF练习 ①
  2. K8S构建1台master2台node+Harbor_笔记
  3. Python 学习笔记(下)
  4. 5分钟完全掌握Python协程
  5. (八)、rm--删除文件或者目录
  6. 80%的学校还在给新生上C语言,它们OUT了吗?
  7. java字符统计+字符串压缩
  8. JavaDailyReports10_05
  9. 风炫安全web安全学习第二十九节课 CSRF防御措施
  10. 使用CSS的clip-path实现图片剪切效果