预处理
C程序的编译预处理用于把每一条C语句用若干条机器指令来实现,生成目标程序。由于#define等编译预处理指令不是C语句,不能被编译程序翻译,需要在真正编译之前作一个预处理,解释完成编译预处理指令,从而把预处理指令转换成相应的C程序段,最终成为由纯粹C语句构成的程序,经编译最后得到目标代码。
C语言的编译预处理处理功能主要包括文件包含(#include)、宏定义(#define)和条件编译。下面介绍条件编译。
条件编译
一般的程序经过编译后,所有的C语句都生成到目标程序中,如果只想把源程序中一部分语句生成目标代码,可以使用条件编译。
对(#ifdef,#else,#endif,#if等)进行说明。以下分3种情况:
1:情况1:
#ifdef _XXXX
...程序段1...
#else
...程序段2...
#endif
这表明如果标识符_XXXX已被#define命令定义过则对程序段1进行编译;否则对程序段2进行编译。
//例子:
#define THING
.............
.............
.............
#ifdef THING
printf("之前THING已经定义过了 \n");
#else
printf("之前THING还没有定义过 \n");
#endif
如果程序开头有#define THING这行,即THING有定义,碰到下面#ifdef THING的时候,当然执行第一个printf。否则第二个printf将被执行。
这种方法可以很方便的开启/关闭整个程序的某项特定功能。
2:情况2:
#ifndef _XXXX
...程序段1...
#else
...程序段2...
#endif
这里使用了#ifndef,表示的是if not def。当然是和#ifdef相反的状况(如果没有定义了标识符_XXXX,那么执行程序段1,否则执行程序段2)。例子就不举了。
3:情况3:
#if 常量
...程序段1...
#else
...程序段2...
#endif
这里表示,如果常量为真(非0,随便什么数字,只要不是0),就执行程序段1,否则执行程序段2。
当需要开启测试的时候,只要将常量变1就好了。而不要测试的时候,只要将常量变0。
C语言的编译预处理
参考C语言的编译预处理
简单分析
1、预处理 -> 头文件展开、宏替换、条件编译、去掉注释
list.i test.i
2、编译 -> 分别编译,检查语法,生成汇编代码
list.s test.s
3、汇编 -> 汇编代码转成二进制的机器码
list.o test.o
4、链接 -> 将两个目标文件链接到一起
1.C程序的过程
预处理 | 处理所有的注释,以空格代替 | gcc -E file.c -o hello.i |
---|---|---|
将所有的#define删除,并且展开所有的宏定义 | ||
处理条件编译指令#if.#ifdef.#elif,#else,#endif | ||
处理#include,展开被包含的文件 | ||
保留编译器需要使用的#pragma指令 | ||
编译 | 对与处理文件进行了一些列词法分析,语法分析和语义分析 | gcc -S file.c -o hello.s |
- 词法分析主要分析关键字,标示符,立即数等是否合法 | ||
- 词法分析主要分析表达式是否遵循语法规则 | ||
-语义分析在语法分析的基础上进一步分析表达式是否合法 | ||
分析结束后进行代码优化生成相应的汇编代码文件 | ||
汇编 | 汇编器将汇编代码转变为机器可以执行的指令 | gcc -c file.s -o hello.o |
- 每个汇编语句几乎都对应一条机器指令 | ||
链接 | 连接器的主要作用是把各个模块之间相互引用的部分处理好,使得各个模块之间能够正确的衔接 |