动态编译和静态编译,共享库
简介
静态函数库
一般扩展名为(.a),这类的函数库通常扩展名为libxxx.a 。
这类函数库在编译的时候会直接整合到程序中,所以利用静态函数库编译成的文件会比较大,这类函数库最大的优点就是编译成功的可执行文件可以独立运行,而不再需要向外部要求读取函数库的内容;但是从升级难易度来看明显没有优势,如果函数库更新,需要重新编译。
动态函数库
动态函数库的扩展名一般为(.so),这类函数库通常名为libxxx.so 。
与静态函数库被整个捕捉到程序中不同,动态函数库在编译的时候,在程序里只有一个“指向”的位置而已,也就是说当可执行文件需要使用到函数库的机制时,程序才会去读取函数库来使用;也就是说可执行文件无法单独运行。一般,在程序第一次执行或者第一次调用的时候,可能会在加载链接库的时候慢点,但这样减少看每个可执行文件的长度, 而且其他的程序如果发现内存中已经存在这个链接库的时候就不需要再次加载了。 另外,使用共享库的可以使用库函数来迭代更新,应用程序将无需更改,只要函数的接口不变。
例子
第1步:编辑得到举例的程序hello.h、hello.c和main.c;
hello.h (见程序1)为该函数库的头文件。
hello.c (见程序2)是函数库的源程序,其中包含公用函数hello,该函数将在屏幕上输出"Hello XXX!"。
main.c (见程序3)为测试库文件的主程序,在主程序中调用了公用函数hello。
--------------------------------------------------------------------------------
程序1: hello.h
#ifndef HELLO_H
#define HELLO_H
void hello(const char *name);
#endif //HELLO_H
--------------------------------------------------------------------------------
程序2: hello.c
#include <stdio.h>
void hello(const char *name)
{
printf("Hello %s!\n", name);
}
--------------------------------------------------------------------------------
程序3: main.c
#include "hello.h"
int main()
{
hello("everyone");
return 0;
}
--------------------------------------------------------------------------------
第2步:将hello.c编译成.o文件
在系统提示符下键入以下命令得到hello.o文件。
# gcc -c hello.c
第3步:由.o文件创建静态库
静态库文件名的命名规范是以lib为前缀,紧接着跟静态库名,扩展名为.a。例如:我们将创建的静态库名为myhello,则静态库文件名就是libmyhello.a。在创建和使用静态库时,需要注意这点。创建静态库用ar命令。在系统提示符下键入以下命令将创建静态库文件libmyhello.a。
# ar cr libmyhello.a hello.o
第4步:在程序中使用静态库
静态库制作完了,如何使用它内部的函数呢?只需要在使用到这些公用函数的源程序中包含这些公用函数的原型声明,然后在用gcc命令生成目标文件时指明静态库名,gcc将会从静态库中将公用函数连接到目标文件中。注意,gcc会在静态库名前加上前缀lib,然后追加扩展名.a得到的静态库文件名来查找静态库文件。在程序3:main.c中,我们包含了静态库的头文件hello.h,然后在主程序main中直接调用公用函数hello。下面先生成目标程序hello,然后运行hello程序看看结果如何。
# gcc -o hello main.c -L. -lmyhello
# ./hello
Hello everyone!
我们删除静态库文件试试公用函数hello是否真的连接到目标文件 hello中了。
# rm libmyhello.a
rm: remove regular file `libmyhello.a'? y
# ./hello
Hello everyone!
程序照常运行,静态库中的公用函数已经连接到目标文件中了。我们继续看看如何在Linux中创建动态库。我们还是从.o文件开始。
第5步:由.o文件创建动态库文件
动态库文件名命名规范和静态库文件名命名规范类似,也是在动态库名增加前缀lib,但其文件扩展名为.so。例如:我们将创建的动态库名为myhello,则动态库文件名就是libmyhello.so。用gcc来创建动态库。在系统提示符下键入以下命令得到动态库文件libmyhello.so。
# gcc hello.c -fPIC -shared -o libmyhello.so
第6步:在程序中使用动态库
在程序中使用动态库和使用静态库完全一样,也是在使用到这些公用函数的源程序中包含这些公用函数的原型声明,然后在用gcc命令生成目标文件时指明动态库名进行编译。我们先运行gcc命令生成目标文件,再运行它看看结果。
# gcc -o hello main.c -L. -lmyhello
# ./hello
./hello: error while loading shared libraries: libmyhello.so: cannot open shared object file: No such file or directory
哦!出错了。快看看错误提示,原来是找不到动态库文件libmyhello.so
。程序在运行时,会在/usr/lib64和/lib
等目录中查找需要的动态库文件。若找到,则载入动态库,否则将提示类似上述错误而终止程序运行。我们将文件libmyhello.so
复制到目录/usr/lib中,再试试。如果不行,配置/etc/ld.so.conf; ldconfig
试下。
# mv libmyhello.so /usr/lib
# ./hello
Hello everyone!
成功了。这也进一步说明了动态库在程序运行时是需要的。
当Linux静态库和Linux动态库同名时, gcc命令将优先使用动态库