gcc

gcc

hello程序的生命周期是从一个高级C语言程序开始的,因为这种形式能够被人读懂。然而,为了在系统上运行hello.c程序,每条C语句都必须被其他程序转化为一系列的低级机器语言指令。然后这些指令按照一种称为可执行目标程序的格式打好包,并以二进制磁盘文件的形式存放起来。目标程序也称为可执行目标文件。Unix系统上,从源文件到目标文件的转化是由编译器驱动程序完成的:

unix> gcc -o hello hello.c

GCC编译器驱动程序读取源程序文件hello.c,并把它翻译成一个可执行目标文件hello。这个翻译的过程可分为四个阶段完成:

image

  • 预处理阶段。预处理器(cpp)根据以字符#开头的命令,修改原始的C程序。比如hello.c中第1行的#include <stdio.h>命令告诉预处理器读取系统头文件stdio.h的内容,并把它直接插入到程序文本中。结果就得到了另一个C程序,通常是以.i作为文件扩展名。

  • 编译阶段。编译器(cc1)将文本文件hello.i翻译成文本文件hello.s,它包含一个汇编语言程序。汇编语言程序中的每条语句都以一种标准的文本格式确切地描述了一条低级机器语言指令。汇编语言是非常有用的,因为它为不同高级语言的不同编译器提供了通用的输出语言。例如,C编译器和Fortran编译器产生的输出文件用的都是一样的汇编语言。

  • 汇编阶段。接下来,汇编器(as)将hello.s翻译成机器语言指令,把这些指令打包成一种叫做可重定位目标程序(relocatable object program)的格式,并将结果保存在目标文件hello.o中。hello.o文件是一个二进制文件,它的字节编码是机器语言指令而不是字符。如果我们在文本编辑器中打开hello.o文件,看到的将是一堆乱码。

  • 链接阶段。请注意,hello程序调用了printf函数,它是每个C编译器都会提供的标准C库中的一个函数。printf函数存在于一个名为printf.o的单独的预编译好了的目标文件中,而这个文件必须以某种方式合并到我们的hello.o程序中。链接器(ld)就负责处理这种合并。结果就得到hello文件,它是一个可执行目标文件(或者简称为可执行文件,可以被加载到内存中,由系统执行。

gcc的组成

部分 描述
c++ gcc的一个版木,默认语言设置为 C++,而且在连接的时候自动包含标准C++库。这和g++一样
ccl 实际的C编译程序
cclplus 实际的C++编泽程序
collect2 在不使用GNU连接程序的系统上,有必要运行collect2来产生特定的全局初始化代码(例如C++的构造函数和析构函数)
configure GCC源代码树根目录中的一个脚木。用于设置配置值和创建GCC编译程序必需的make程序的描述文件
crt0.o 这个初始化和结束代码是为每个系统定制的,而且也被编译进该文件,该文件然后会被连接到每个可执行文件中来执行必要的启动和终止程序
cygwin1.dll Windows的共享库提供的API,模拟UNIX系统调用
f77 该驱动程序可用于编译Fortran
f771 实际的Fortran编译程序
g++ gcc的一个版木,默认语言设置为C++,而且在连接的时候自动包含标准C++库。这和c++一样
gcc 该驱动程序等同于执行编译程序和连接程序以产生需要的输出
gcj 该驱动程序用于编译 Java
gnat1 实际的Ada编译程序
gnatbind 一种工具,用于执行Ada语言绑定
gnatlink 一种工具,用于执行Ada语言连接
jc1 实际的Java编译程序
libgcc 该库包含的例程被作为编泽程序的一部分,是因为它们可被连接到实际的可执行程序中。它们是特殊的例程,连接到可执行程序,来执行基木的任务,例如浮点运算。这些库中的例程通常都是平台相关的
libgcj 运行时库包含所有的核心Java
libobjc 对所有Objective-C程序都必须的运行时库
libstdc++ 运行时库,包括定义为标准语言一部分的所有的C++类和函数