以最简单的hello.c程序文件为例 :
#include <stdio.h>
#include <stdlib.h>
int main()
{
printf("hello world!\n");
return 0;
}
【 第一步】 预处理 hello.c—hello.i
预处理过程实质上是处理’’#’’,将#include包含的头文件直接拷贝到hello.c当中;将#define定义的宏进行替换,同时将代码中没用的注释部分删除等。具体做的事儿如下:
- 将所有的#define删除,并展开所有的宏定义。
- 处理所有的条件编译指令,#ifdef #ifndef #endif等,就是带#的那些。
- 处理#include,将#include指向的文件插入到该行处。
- 删除所有注释。
- 添加行号和文件标示 ,这样在调试和编译出错的时候才知道是哪个文件的哪一行。
- 保留#pragma编译器指令,因为编译器需要使用它们。
【第二步】编译 hello.i—hello.s
编译的过程实质上是把高级语言翻译成机器语言的过程。
- 词法分析
- 语法分析
- 语义分析
- 优化后生成相应的汇编代码。
- 从高级语言->汇编语言->机器语言(二进制)
【第三步】汇编 hello.s—hello.o
汇编器as将hello.s翻译成机器语言保存在hello.o中(二进制文本形式)。
【第四步】链接 hello.o— hello
就像刚才的hello.c,它使用到了C标准库的东西“printf”,但是编译过程只是把源文件翻译成二进制而已,这个二进制还不能直 接执行,这个时候就需要做一个动作, 将翻译成的二进制与需要用到的库绑定在一块。
也可以直接使用以下命令完成编译gcc hello.c -o hello 可以生成可执行程序。即gcc不带任何参数。ldd可以看到你的可执行程序依赖的库。