话说ISOC99有自己的一系列标准C函数库,例如我们熟悉的libc.a(包含标准I/O函数、字符串操作函数和整数数学函数)和libm.a(浮点数数学函数),可供我们在使用gcc编译工具编译程序时调用。那么,如果我们在日常学习或项目开发中积累了许多好的函数,希望日后在其他项目中能够复用时,我们又该如何去保存他们呢?今天我就告诉大家怎么办?
1. 看gcc编译器都干了些什么?
所谓,知己知彼,百战不殆。
首先,用一个例子带大家了解一下gcc的编译过程。
//main.c
int sum (int *a, int n);
int array[2] = {1, 2};
int main()
{
int val = sum(array, 2);
return val;
}
//sum.c
int sum(int *a, int n)
{
int i, s = 0;
for(i=0; i < n; i++){
s += a[i];
}
return s;
}
假如一个项目共有以上2个源程序组成。main函数初始化一个整数数组,然后调用一个定义在另一个文件中的sum函数来对数组元素求和。
一般,我们会直接在命令行下键入gcc -Og -o prog main.c sum.c
来直接编译出二进制的可执行的ELF格式的目标文件prog。那在此之间,gcc到底都干了些什么工作呢?
(备注:可以在gcc命令中添加参数-v观察到工作过程。)
看图☞
1.1 cpp预处理
cpp main.c -o main.i
预处理器cpp将源代码main.c翻译成一个ASCII码的中间文件main.i。
1.2 ccl编译
ccl -Og main.i -o main.s
C编译器ccl将中间文件翻译成一个ASCII码的汇编语言文件main.s。
备注:-O参数选择优化级别,由低到高分别为g<1<2<3。一般g和1优化级别用于调试阶段,2和3用于最后交付前的优化阶段。
1.3 as汇编
as main.s -o main.o
汇编器as将汇编文件翻译成一个二进制的可重定位的目标文件main.o。
###1.4处理其他源文件
经过相同的步骤,将另一个源文件sum.c转换成sum.o
1.5 ld链接
ld -o prog main.o sum.o
1.6 加载运行
以上步骤都做完之后,最后在命令行(shell)键入./prog
就可以运行啦。shell会调用操作系统中一个叫加载器的函数,它将可执行文件中的代码和数据复制到内存,然后将控制权交给该程序。
2. 创建自己的函数库
假设在某个项目中你自己编写了两个函数addvec() 和mulvec(),感觉很牛逼很好用,希望以后在其他项目里也能用的上。这里有两种方法,一是保存他们的源代码,以后编写程序时插进去;二是将他们加入到一个自己的私有库中,在使用时可在其他文件中直接调用。
前者or后者哪个方便?
如果你选择前者,好的你可以结束阅读课,选择后者,请继续往下看:
###2.1 制作目标文件
gcc -c myfuncs.c
将包含你所需要打包的函数的c文件编译成可重定位的目标文件
2.2生成静态库
ar rcs libmyfuncs.a myfuncs.o
利用AR库制作工具,创建我们所需的静态库。
2.3 制作.h头文件
新建一个头文件myfuncs.h,要求其中包含所有myfuncs.a库中函数的原型语句。
3. 使用方法
假设我们在示例程序main2.c中需要使用静态库libvector.a中的函数。我们该怎么办呢?
3.1 添加头文件;
将头文件vector.h添加到自己的工程目录中,并在main2.c文件的顶端添加#include "vector.h"
语句,之后在后续的编程语句中就可以调用库中的函数了(假设我们需要调用库中的addvec()函数)。
3.2 正确使用编译选项
当你的程序main2.c编写完毕后,需要在命令行链接我们的自定义静态库libvector.a。
gcc -c main2.c
gcc -static -o prog2c main2.o -L. -lvector
或者
gcc -c main2.c
gcc -static -o prog2c main2.o ./libvector.a
- 第一条语句用来生成目标文件main2.o;
- 第二条语句中,
- -static参数告诉连接器应该构建一个完全链接的(所有函数和全局变量的地址均已添加进ELF格式的文件了)可执行目标文件,它可以直接加载到内存并运行,且在加载时无需再做其他链接。
- -L 告诉链接器应该在当前目录下查找库(因为L后面跟的是一个点,而点在linux系统中代表当前目录)。
- -l 后面直接跟你需要链接的库的名字的缩写myfuncs,而不是全称libmyfuncs.a。
- 切记:链接器是从左到右依次解析各个目标文件的,并把在各个目标文件中找到的所有外部应用符号记录下来,最后在标准C库文件和命令行上指定的库文件中寻找定义。如果都能找到,它就合并及重定位所有目标文件,最终构建输出可执行文件。如果有任一个符号未找到定义,就输出错误并终止。所以,被依赖的库文件一定要位于引用它文件的后面!!!如果有循环依赖,那么就在命令行上多写几次。
假设tmp.c调用libx.a中的函数,该库又调用liby.a中的函数,而liby.a又调用libx.a中的函数。那么,libx.a必须在命令行上重复出现2次:
gcc tmp.c libx.a liby.a libx.a
- 注意:链接器还会默认链接C标准函数库libc.a。
3.3 全局过程
=我是华丽的分割线=
更多知识:
***点击关注专题:***嵌入式Linux&ARM
***或浏览器打开:***https://www.jianshu.com/c/42d33cadb1c1