(1). 首先调用cpp进行预处理,在预处理过程当中,对源代码文件中的文件包含(include)、预编译语句(如宏定义define等)进行分析。 linux
(2). 接着调用cc1进行编译,这个阶段根据输入文件生成以.o为后缀的目标文件。 小程序
(3). 汇编过程是针对汇编语言的步骤,调用as进行工做,通常来说,.S为后缀的汇编语言源代码文件和汇编、.s为后缀的汇编语言文件通过预编译和汇编以后都生成以.o为后缀的目标文件。 bash
(4). 当全部的目标文件都生成以后,gcc就调用ld来完成最后的关键性工做,这个阶段就是链接。在链接阶段,全部的目标文件被安排在可执行程序中的恰当的位置,同时,该程序所调用到的库函数也从各自所在的档案库中连到合适的地方。 服务器
基本概念 函数
#include <>#include <>int main(){sayhello();return 0;}
#include <>void sayhello(){printf("hello,world ");}
三 、 示例 post
在建立函数库前,咱们先来准备举例用的源程序,并将函数库的源程序编译成.o文件。 测试
第1步:编辑获得举例的程序--hello.h、hello.c和main.c; spa
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文件;
不管静态库,仍是动态库,都是由.o文件建立的。所以,咱们必须将源程序hello.c经过gcc先编译成.o文件。
在系统提示符下键入如下命令获得hello.o文件。
# gcc -c hello.c
#
(注1:本文不介绍各命令使用和其参数功能,若但愿详细了解它们,请参考其余文档。)
(注2:首字符"#"是系统提示符,不须要键入,下文相同。)
咱们运行ls命令看看是否生存了hello.o文件。
# ls
hello.c hello.h hello.o main.c
#
(注3:首字符不是"#"为系统运行结果,下文相同。)
在ls命令结果中,咱们看到了hello.o文件,本步操做完成。
下面咱们先来看看如何建立静态库,以及使用它。
--------------------------------------------------------------------------------
第3步:由.o文件建立静态库;
静态库文件名的命名规范是以lib为前缀,紧接着跟静态库名,扩展名为.a。例如:咱们将建立的静态库名为myhello,则静态库文件名就是libmyhello.a。在建立和使用静态库时,须要注意这点。建立静态库用ar命令。
在系统提示符下键入如下命令将建立静态库文件libmyhello.a。
# ar cr libmyhello.a hello.o
#
咱们一样运行ls命令查看结果:
# ls
hello.c hello.h hello.o libmyhello.a main.c
#
ls命令结果中有libmyhello.a。
--------------------------------------------------------------------------------
第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 -shared -fPCI -o libmyhello.so hello.o
#
咱们照样使用ls命令看看动态库文件是否生成。
# ls
hello.c hello.h hello.o libmyhello.so main.c
#
--------------------------------------------------------------------------------
第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/lib和/lib等目录中查找须要的动态库文 件。若找到,则载入动态库,不然将提示相似上述错误而终止程序运行。咱们将文件libmyhello.so复制到目录/usr/lib中,再试试。
# mv libmyhello.so /usr/lib
# ./hello
Hello everyone!
#
成功了。这也进一步说明了动态库在程序运行时是须要的。
咱们回过头看看,发现使用静态库和使用动态库编译成目标程序使用的gcc命令彻底同样,那当静态库和动态库同名时,gcc命令会使用哪一个库文件呢?抱着对问题必究到底的心情,来试试看。
先删除 除.c和.h外的 全部文件,恢复成咱们刚刚编辑完举例程序状态。
# rm -f hello hello.o /usr/lib/libmyhello.so
# ls
hello.c hello.h main.c
#
在来建立静态库文件libmyhello.a和动态库文件libmyhello.so。
# gcc -c hello.c
# ar cr libmyhello.a hello.o
# gcc -shared -fPCI -o libmyhello.so hello.o
# ls
hello.c hello.h hello.o libmyhello.a libmyhello.so main.c
#
经过上述最后一条ls命令,能够发现静态库文件libmyhello.a和动态库文件libmyhello.so都已经生成,并都在当前目录中。而后,咱们运行gcc命令来使用函数库myhello生成目标文件hello,并运行程序 hello。
# 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
#
从程序hello运行的结果中很容易知道,当静态库和动态库同名时, gcc命令将优先使用动态库。