参考链接:https://blog.youkuaiyun.com/qq_24990383/article/details/81706127
一、 预处理
- C程序的编译预处理用于把每一条C语句用若干条机器指令来实现,生成目标程序。由于#define等编译预处理指令不是C语句,不能被编译程序翻译,需要在真正编译之前作一个预处理,解释完成编译预处理指令,从而把预处理指令转换成相应的C程序段,最终成为由纯粹C语句构成的程序,经编译最后得到目标代码。
- C语言的编译预处理处理功能主要包括文件包含(#include)、宏定义(#define)和条件编译。下面介绍条件编译。
二、条件编译
一般的程序经过编译后,所有的C语句都生成到目标程序中,如果只想把源程序中一部分语句生成目标代码,可以使用条件编译。对(#ifdef,#else,#endif,#if等)进行说明。以下分3种情况:
情况一
#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将被执行。
- 这种方法可以很方便的开启/关闭整个程序的某项特定功能。
情况二
#ifndef _XXXX
...程序段1...
#else
...程序段2...
#endif
这里使用了#ifndef,表示的是if not def。当然是和#ifdef相反的状况(如果没有定义了标识符_XXXX,那么执行程序段1,否则执行程序段2)。
情况三
#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、链接 -> 将两个目标文件链接到一起
四、C程序编译过程
阶段 | 说明 | 命令 |
---|---|---|
预处理 | 处理所有的注释,以空格代替 | gcc -E file.c -o file.i |
将所有的#define删除,并且展开所有的宏定义 | ||
处理条件编译指令#if.#ifdef.#elif,#else,#endif | ||
处理#include,展开被包含的文件 | ||
保留编译器需要使用的#pragma指令 | ||
编译 | 对与处理文件进行了一些列词法分析,语法分析和语义分析 | gcc -S file.c -o file.s |
词法分析主要分析关键字,标示符,立即数等是否合法 | ||
词法分析主要分析表达式是否遵循语法规则 | ||
语义分析在语法分析的基础上进一步分析表达式是否合法 | ||
分析结束后进行代码优化生成相应的汇编代码文件 | ||
汇编 | 汇编器将汇编代码转变为机器可以执行的指令 | gcc -c file.s -o file.o |
每个汇编语句几乎都对应一条机器指令 | ||
连接器的主要作用是把各个模块之间相互引用的部分处理好,使得各个模块之间能够正确的衔接 |
五、GCC编译器命令:区分大小写
1. gcc -E : 只做预处理,其它都不做,(.i)
2. gcc -S : 检查语法错误等,生成汇编文件(.s)
3. gcc -c : 完成汇编,不链接(.o)
4. gcc -o : 修改目标文件名
5. gcc : 编译+链接,生成可执行文件,默认是 a.out
6. gcc -Wall : 生成所有的警告信息(提示代码质量,少警告)
7. Gcc -D宏名: 条件编译,用于版本选项开关