linux gcc 的编译过程 详解

来源:互联网 发布:it初学者 编辑:程序博客网 时间:2024/05/16 01:50

gcc的编译过程分为四步,分别为:
(1)预编译 (Pre-Processing)
(2)编译 (Compiling)
(3)汇编 (Assembling)
(4)链接 (Linking)
以hello.c为例说明:
#include<stdio.h>
int main(void)
{
 printf("Hello World!");
 return 0;
}

(1)预编译阶段 (Pre-Processing)
  在该阶段,编译器将上述代码中的stdio.h编译进来,并且使用gcc -E进行查看(-E让gcc在预处理结束后停止编译)
  gcc -E hello.c -o hello.i
  查看hello.i可见,gcc确实进行了预处理,它把"stdio.h"的内容插入到hello.i文件中
(2)编译阶段
  在这个阶段,gcc首先检查代码的规范性,是否有语法错误等,以确定代码的实际要做的工作,在检查无误后
  gcc把代码编译成汇编语言,可以使用-S选项进行查看,该选项之进行编译而不进行汇编,生成汇编代码
  gcc -S hello.i -o hello.s
(3)汇编阶段
 就是把编译阶段生成的.s文件转成目标文件,可以使用选项-c ,就是把汇编代码转化成.o的二进制目标代码了
 gcc -c hello.s -o hello.o

(4)链接阶段
 在成功编译之后就进入了链接阶段,在这里涉及到一个重要的概念:函数库。
 读者重新查看这个小程序,在该小程序中没有并没有定义"printf"的函数实现,且在预编译中包含进的"stdio.h"
 中也只有该函数的声明,而没有定义该函数的实现,那么在哪实现"printf"函数哪?,最后答案是:系统把这些函数实现
 都被做到名为libc.so.6的库文件中去了,在没有特别指定时gcc会到系统默认的搜索路径"/usr/lib"下去查找
 也就是链接到libc.so.6库函数中去,这样就实现了"printf"了,而这也是链接的作用
 函数库一般分为静态库和动态库两种,静态库是指在编译链接时,把库文件的代码全部加到可执行程序中去,因此生成的
 可执行文件比较大,在运行时就不再需要库文件了,后缀名为.a。动态库与之相反,在编译链接时并没有把库文件的代码
 加入到可执行文件中,而是在程序执行时由运行时链接文件加载库,这样可以节省系统的开销,动态库一般后缀名为.so。
 完成链接后gcc就可以生成可执行文件

gcc hello.o -o hello