一般情况下,源程序中所有的行都参加编译。但是有时希望对其中一部分内容只在满足一定条件下才进行编译,即对一部分内容指定编译条件,这就是“条件编译”。(conditional compile)
条件编译语句排版时,需考虑以下三种位置:
(1)条件编译语句块与函数定义体之间不存在相互嵌套(主要在(.h)文件中)
◆ 条件编译关键字语句顶格左对齐;
◆ 所含的#include语句(块) #define语句(块)甚至是被嵌套下级条件编译语句块,按照语句块嵌套的排版方式进行缩进排版 。
(2)条件编译语句块嵌套在函数体之外(主要在(.c)文件中)
这种情况下,条件编译语句块不影响函数体
◆ 条件编译关键字语句顶格左对齐;
◆ 所含的函数体定义无需缩进,依旧按照单个函数体定义的排版方式进行。
(3)条件编译语句嵌套在函数体内 (主要在(.c)文件中)
a)当条件编译语句块与被包语句所属的语句块之间没有逻辑路径交叉时,以下两种方式均可
◆ 按照语句块嵌套方式进行缩进排版 (推荐);
◆ 条件编译语句不影响原先语句块排版,条件编译语句与所包含的关键字语句块左对齐 。
b)当条件编译语句块与被包语句所属的语句块之间存在逻辑路径交叉时
◆ 条件编译语句顶格左对齐,其它语句按照正常顺序排版。
条件编译的形式如下所示(NNN、MMM等都是在某处已经定义为 1 或者 0 的):
#if NNN
statement1;
#elif MMM
statement2;
#else
statement3;
#endif
条件编译指令将决定那些代码被编译,而哪些是不被编译的。可以根据表达式的值或者某个特定的宏是否被定义来确定编译条件。
1.#if指令
#if指令检测跟在制造另关键字后的常量表达式。如果表达式为真,则编译后面的代码,直到出现#else、#elif或#endif为止;否则就不编译。
2.#endif指令
#endif用于终止#if预处理指令。
#define DEBUG 0
main()
{
#if DEBUG
printf("Debugging/n");
#endif
printf("Running/n");
}
由于程序定义DEBUG宏代表0,所以#if条件为假,不编译后面的代码直到#endif,所以程序直接输出Running。
如果去掉#define语句,效果是一样的。
3.#ifdef和#ifndef
#define DEBUG
main()
{
#ifdef DEBUG
printf("yes/n");
#endif
#ifndef DEBUG
printf("no/n");
#endif
}
#if defined等价于#ifdef; #if !defined等价于#ifndef
4.#else指令
#else指令用于某个#if指令之后,当前面的#if指令的条件不为真时,就编译#else后面的代码。#endif指令将中指上面的条件块。
#define DEBUG
main()
{
#ifdef DEBUG
printf("Debugging/n");
#else
printf("Not debugging/n");
#endif
printf("Running/n");
}
5.#elif指令
#elif预处理指令综合了#else和#if指令的作用。
#define TWO
main()
{
#ifdef ONE
printf("1/n");
#elif defined TWO
printf("2/n");
#else
printf("3/n");
#endif
}
程序很好理解,最后输出结果是2。
6.其他一些标准指令
#error指令将使编译器显示一条错误信息,然后停止编译。
#line指令可以改变编译器用来指出警告和错误信息的文件号和行号。
#pragma指令没有正式的定义。编译器可以自定义其用途。典型的用法是禁止或允许某些烦人的警告信息。