1. 什么是gcc?
  gcc的全称是GNU Compiler Collection,它是一个能够编译多种语言的编译器。开始gcc是作为C语言的编译器(GNU C Compiler),现在除了c语言,还支持C++、java、Pascal等语言。gcc支持多种硬件平台。
  2. gcc的特点
  gcc是一个可移植的编译器,支持多种硬件平台。例如ARM、X86等等。
  gcc不仅是个本地编译器,它还能跨平台交叉编译。所谓的本地编译器,是指编译出来的程序只能够在本地环境进行运行。而gcc编译出来的程序能够在其他平台进行运行。例如嵌入式程序可在x86上编译,然后在arm上运行。
  gcc有多种语言前端,用于解析不同的语言。
  gcc是按模块化设计的,可以加入新语言和新CPU架构的支持。
  gcc是自由软件。任何人都可以使用或更改这个软件。
  3. gcc编译程序的过程
  gcc编译程序主要经过四个过程:
  预处理(Pre-Processing)
  编译 (Compiling)
  汇编 (Assembling)
  链接 (Linking)

  预处理实际上是将头文件、宏进行展开。编译阶段,gcc调用不同语言的编译器,例如c语言调用编译器ccl。gcc实际上是个工具链,在编译程序的过程中调用不同的工具。汇编阶段,gcc调用汇编器进行汇编。链接过程会将程序所需要的目标文件进行链接成可执行文件。汇编器生成的是可重定位的目标文件,学过操作系统,我们知道,在源程序中地址是从0开始的,这是一个相对地址,而程序真正在内存中运行时的地址肯定不是从0开始的,而且在编写源代码的时候也不能知道程序的地址,所以重定位能够将源代码的代码、变量等定位为内存具体地址。下面以一张图来表示这个过程,注意过程中文件的后缀变化,编译选项和这些后缀有关。
  这是GCC编译的四个步骤。
  4. gcc常用选项
  来看一下gcc常用选项

  现在我们有源文件hello.c,下面是一些gcc的使用示例:
  gcc -E hello.c -o hello.i   对hello.c文件进行预处理,生成了hello.i 文件
  gcc -S hello.i -o hello.s    对预处理文件进行编译,生成了汇编文件
  gcc -c hello.s -o hello.o  对汇编文件进行编译,生成了目标文件
  gcc hello.o -o hello 对目标文件进行链接,生成可执行文件
  gcc hello.c -o hello 直接编译链接成可执行目标文件
  gcc -c hello.c 或 gcc -c hello.c -o hello.o 编译生成可重定位目标文件
  使用gcc时可以加上-Wall选项。下面这个例子如果不加上-Wall选项,编译器不会报出任何错误或警告,但是程序的结果却不是预期的:
  //bad.c
  #include<stdio.h>
  int main()
  {
  printf("the number is %f ",5);  //程序输出了the number is 0.000000,结果错误
  return 0;
  }
  使用-Wall选项:
  gcc -Wall bad.c -o bad
  gcc将输出警告信息:
  warning: format ‘%f’ expects argument of type ‘double’, but argument 2 has type ‘int’ [-Wformat=]
  printf("the number is %f ",5);
  5. gcc编译多个文件
  假设现在有三个文件:hello.c hello.h main.c ,三个文件的内容如下:
// hello.c
#include<stdio.h>
#include"hello.h"
void printHello()
{
printf("hello world! ");
}
//main.c
#include<stdio.h>
#include"hello.h"
int main()
{
printHello();
return 0;
}
//hello.h
//仅包含函数声明
#ifndef _HELLO_
#define _HELLO_
void printHello();
#endif
  编译这三个文件,可以一次编译:
  gcc hello.c main.c -o main 生成可执行文件main
  也可以独立编译:
  gcc -Wall -c main.c -o main.o
  gcc -Wall -c hello.c -o hello.o
  gcc -Wall main.o hello.o -o main
  独立编译的好处是,当其中某个模块发送改变时,只需要编译该模块行,不必重新编译所有文件,这样可以节省编译时间。