预处理:不会检查错误 头文件替换 删除注释 条件编译
gcc -E hello.c -o hello.i
(1)将所有的#define删除,并展开所有的宏定义。说白了就是字符替换。
(2)处理所有的条件编译(预编译)指令,#ifdef #ifndef #endif等,就是带#的那些。
(3)条件语句中符合判断条件的会保留,不符合的会删除。
(3)处理#include预编译指令,将#include指向的文件插入到该行处(预编译指令的位置)。
(4)删除所有注释。
(5)添加行号和文件标示,这样的在调试和编译出错的时候才知道是是哪个文件的哪一行,便于调试。
(6)保留#pragma编译器指令,因为编译器需要使用它们。
(7)生成(.i)临时文件(包括去注释、宏替换、头文件展开、条件编译),编译生成的文件不包含任何宏定义,因为宏已经被展开,并且包含的文件已经被插入到.i文件中。
编译:检查错误 生成汇编文件 实质是把高级语言编译成机器可识别的汇编语言
gcc -S hello.i -o hello.s
(1)扫描、语法分析、语义分析、源代码分析、目标代码生成、目标代码优化。
(2)生成汇编代码。
(3)汇总符号。
(4)生成.s文件。
汇编: 将汇编文件生成二进制文件
gcc -c hello.s -o hello.o
(1)根据汇编指令和特定平台,把汇编指令翻译成二进制形式。
(2)合并各个section,合并符号表。
(3)生成.o文件。
链接:将二进制文件生成可执行文件
gcc hello.o -o hello
(1)链接器把目标文件与所需要的附加的目标文件(如静态链接库、动态链接库)链接起来成为可执行的文件。
(2)合并各个.obj文件的section,合并符号表,进行符号解析。
(3)符号地址重定位。
(4)生成可执行文件。