很多场景下,都会引用三方的库,同时也可能自己编写相关的库,提供能力。
三方库相关编译指令:
gcc -l
-l 参数就是用来指定程序要链接的库,-l 参数紧接着就是库名,那么库名跟真正的库文件名有什么关系呢?就拿数学库来说,他的库名是 m,他的库文件名是 libm.so,很容易看出,把库文件名的头lib 和尾 .so 去掉就是库名了。好了现在我们知道怎么得到库名, 当我们自已要用到一个第三方提供的库名字 libtest.so,那么我们只要把 libtest.so 拷贝到 /usr/lib 里,编译时加上 -ltest 参数,我们就能用上 libtest.so 库了(当然要用 libtest.so 库里的函数,我们还需要与 libtest.so 配套的头文件)
gcc -L
放在/lib 和 /usr/lib 和 /usr/local/lib里的库直接用-l参数就能链接了,但如果库文件没放在这三个目录里,而是放在其他目录里,这时我们只用-l参数的话,链接还是会出错,出错信息大概是:“/usr/bin/ld: cannot find -lxxx”,也就是链接程序 ld 在那3个目录里找不到libxxx.so。这时另外一个参数-L就派上用场了,比如常用的X11 的库,它在 /usr/X11R6/lib 目录下,我们编译时就要用 -L/usr/X11R6/lib -lX11 参数,-L 参数跟着的是库文件所在的目录名。再比如我们把libtest.so 放在/aaa/bbb/ccc 目录下,那链接参数就是 -L/aaa/bbb/ccc -ltest
gcc -I
-I 参数是用来指定头文件目录,/usr/include 目录一般是不用指定的,gcc 知道去那里找,但是如果头文件不在 /usr/include 里我们就要用 -I 参数指定了,比如头文件放在/myinclude 目录里,那编译命令行就要加上 -I/myinclude 参数了,如果不加你会得到一个 “xxxx.h: No such file or directory” 的错误。-I 参数可以用相对路径,比如头文件在当前目录,可以用 -I. 来指定。
gcc –shared
-shared该选项指定生成动态连接库(让连接器生成T类型的导出符号表,有时候也生成弱连接W类型的导出符号),不用该标志外部程序无法连接。相当于一个可执行文件
gcc -fPIC
-fPIC:表示编译为位置独立的代码,不用此选项的话编译后的代码是位置相关的所以动态载入时是通过代码拷贝的方式来满足不同进程的需要,而不能达到真正代码段共享的目的。
本文以linux为例,进行相关的示例讲解。
假设一个场景: 程序员A 编写一个utils库,对常用的函数进行封装,程序员B在开发中进行使用。
utils库代码示例:
utils.h
#ifndef __UTILS_H__
#define __UTILS_H__
void printBin(int num);
double factorial(unsigned int i);
#endif
utils.c
void printBin(int num)
{
if(num)
printBin(num/2);
else
return;
printf("%d",num%2);
}
double factorial(unsigned int i)
{
if(i <= 1)
{
return 1;
}
return i * factorial(i - 1);
}
然后进行库编译:
gcc -fPIC -shared -o libutils.so utils.c
程序员B具体使用示例:
test.c
#include
#includeint main(int argc, char *argv[])
{
int a = 10;
printf("%d的二进制为:",a);
printBin(a);
printf("\n");
printf("%d 的阶乘为 %f\n", a, factorial(a));
return 0;
}
然后进行编译,有两种方式进行编译引用。
1. 将头文件与库放入库目录中。
linux c 标准库文件存放的路径为: /lib 和 /usr/lib
linux c 标准头文件存放的路径为: /usr/include
2. 使用链接方式
gcc test.c -o /test -I ./-L ./ -lutils
运行检查:
执行./test
10的二进制为:1010
10 的阶乘为 3628800.000000