动态库和静态库的介绍
一般情况下,在项目里会把功能相似的代码封装成库,方便使用和管理,同时增加了代码的内聚性。库分为两种,一种为静态库,文件名以.a结尾,另一种是动态库,文件名以.so结尾。静态库和动态库的使用各有利弊。使用静态库的好处就是简单,因为在链接的时候直接把静态库中的内容链接到可执行文件中,链接完成后,就不再需要静态库了。静态库的缺点是,增大了可执行文件的体积,增长了可执行文件装载的时间。同时静态库不能共享,所有使用静态库的可执行文件中,都有静态库中的内容。动态库则没有这些缺点,因为动态库是在程序开始运行后才载入的,而且程序可以根据需求自主选择载入动态库的时机,所以使用动态库可以加快可执行文件装载的时间。同时,动态库是可以共享的,多个程序在运行中可以使用同一个动态库,可以减少程序总的大小。
现在用一个微型的工程,来讲述静态库、动态库的生成和使用。假设有3个.c文件,分别为:add.c sub.c main.c,实现了add()和sub()这两个函数,功能是返回两个整数相加或者相减的结果,main.c中使用到了这两个函数。下面讲述怎样把这add.c和sub.c两个文件生成并使用静态库和动态库,然后生成可执行文件。
//add.h:
int add(int a, int b);
//add.c:
int add(int a, int b) {
return a + b;
}
//sub.h:
int sub(int a, int b);
//sub.c:
int sub(int a, int b) {
return a - b;
}
//main.c:
#include "add.h"
#include "sub.h"
#include <stdio.h>
int main() {
printf("Result of 1 + 1 is: %d\n", add(1,1));
printf("Result of 6 - 3 is: %d\n", sub(6,3));
return 0;
}
一、生成静态库
生成静态库一般使用ar命令,关于ar命令详情可参考这里,或者Linux Manual Page。
通常步骤是把生成静态库所涉及到的.c文件全都编译,生成.o文件,然后使用ar命令生成.a文件,即静态库。这里需要使用到gcc的-c选项,意思是只进行编译,不进行链接。
gcc -W -c add.c -o add.o
gcc -W -c sub.c -o sub.o
ar cvr libaddsub.a add.o sub.o
此时得到了静态库libaddsub.a。可以用命令nm来查看.a中的符号名称。
二、使用静态库
使用静态库的方法非常简单,直接把静态库的名字加入到需要编译的文件里就可以。如下所示,注意被依赖的静态库应该放在后面:
gcc -W -c main.c -o main.o
gcc main.o libaddsub.a -o test
执行test得到如下结果:
Result of 1 + 1 is: 2
Result of 6 - 3 is: 3
生成动态库要求编译出的.o文件是位置无关的代码,也就是说需要使用-fpic选项。然后使用gcc的-share选项生成动态库。
gcc -W -c add.c -fpic -o add.o
gcc -W -c sub.c -fpic -o sub.o
gcc -shared add.o sub.o -o libaddsub.so
生成可执行文件的时候需要指定动态库所在的位置,以及动态库的名称。-L选项表示指定动态库所在的目录,-l选项表示需要链接的动态库的名称。一般Linux下动态库的明明规则都是libxxx.so,其中xxx为动态库的真正名称,所以-l选项的写法是-lxxx。-L选项与它后的内容以及-l选项与它后面的内容之间可以没有空格,也可以加上空格。也就是说,-lxxx和-l xxx都是可以的。
gcc -W -c main.c -o main.o
gcc main.o -o test -L./ -laddsub
如果此时直接运行test的话,会得到类似下面的错误:
./test: error while loading shared libraries: libaddsub.so: cannot open shared object file: No such file or directory 意思是libaddsub.so不在系统默认加载动态库的目录里面。解决的办法有很多种:
(1)系统默认的动态库目录为/lib*,/usr/lib*,和/etc/ld.so.conf文件中配置的目录。所以可以把libaddsub.so复制到默认的动态库加载目录里,比如/usr/lib目录下。一般安装程序时会这么做,安装完后程序就可以正常启动了。
(2)修改/etc/ld.so.conf配置文件。把所需的动态库所在的目录添加到这个文件里。
(3)启动程序之前向环境变量LD_LIBRARY_PATH中加入动态库所在的目录。可以使用export,或者这样:
LD_LIBRARY_PATH=$LD_LIBRARY_PATH:PWD ./test
这只是一个临时的做法,不建议使用这个方法。
(4)生成可执行文件时加入-Wl,-rpath=xxx参数,其中xxx是动态库所在的目录,这里最好写成绝对路径。-Wl表示后面的参数是链接器ld的参数而不是gcc的参数。-rpath表示运行时加载动态库的目录。这样做的话就可以直接运行可执行文件了,因为动态库所在目录的信息已经保存在了可执行文件中。例如:
gcc main.o -o test -L./ -laddsub -Wl,-rpath=./
./test
运行程序后,还是可以得到和使用静态库相同的结果。