1 被隐藏了的过程
1.1 预编译
首先是源代码文件hello.c
和相关的头文件,如stdio.h
等被预编译器cpp预编译成一个.i
文件。对于C++程序来说,它的源代码文件的扩展名可能是.cpp
或.cxx
,头文件的扩展名可能是.hpp
,而预编译后的文件扩展名是.ii
。第一步预编译的过程相当于如下命令(-E
表示只进行预编译):
$gcc -E hello.c -o hello.i
或者:
$cpp hello.c > hello.i
预编译过程主要处理那些源代码文件中的以#
开始的预编译指令。比如#include
、#define
等,主要处理规则如下:
- 将所有的
#define
删除,并且展开所有的宏定义。处理所有条件预编译指令,比如#if
、#ifdef
、#elif
、#else
、#endif
。 - 处理
#include
预编译指令,将被包含的文件插入到该预编译指令的位置。注意,这个过程是递归进行的,也就是说被包含的文件可能还包含其他文件。 - 删除所有的注释
//
和/**/
- 添加行号和文件名标识,比如
#2 “hello.c” 2
,以便于编译时编译器产生调试用的行号信息及用于编译时产生编译错误或警告时能够显示行号。 - 保留所有的
#pragma
编译器指令,因为编译器须要使用它们。
经过预编译后的.i
文件不包含任何宏定义,因为所有的宏已经被展开,并且包含的文件也已经被插入到.i
文件中。所以当我们无法判断宏定义是否正确或头文件包含是否正确时,可以查看预编译后的文件来确定问题
1.2 编译
1.3 汇编
1.4 链接
2 编译器做了什么
2.1 词法分析
2.2 词法分析
2.3 语义分析
2.4 中间语言生成
2.5 目标代码生成与优化
3 链接器年龄比编译器长
4 模块拼装-静态链接