无值宏定义
概念: 该宏的定义语句中不需要给宏指定指定的值,因此它在代码中并没有实际的意义,一般用于条件编译的判断语句。
#define BIG_ENDIAN
#define __cplusplus
条件编译
- 概念:有条件的编译,通过控制某些宏的值(或某个宏是否有定义),来决定编译哪段代码。
- 基本语法:
#if A // 判断 宏 A 它的值是否为真 ... // 如果 MACRO 为真,那么该段代码将被编译,否则被丢弃 #endif // C 中的条件判断语句 if( 布尔表达式 ) { // 根据条件是否需要执行本代码块 }
- 形式:
- 形式1:判断表达式 MACRO 是否为真,据此决定其所包含的代码段是否要编译
#define DEBUG 100
// 根据宏 DEBUG 的值来决定是否编译 第 14行代码 非零 则真
#if DEBUG
printf("__%d__%s__%s__\n" , __LINE__ , __FUNCTION__ , __FILE__ );
#endif
-
- 形式2:判断某一个宏是否被定义了或是否没有被定义,据此决定其所包含的代码段是否要编译
// 判断宏 DEBUG 是否有定义, 如果有则以下代码是生效的,否则不生效 #ifdef DEBUG printf("__%d__%s__%s__\n" , __LINE__ , __FUNCTION__ , __FILE__ ); #endif // 判断宏 DEBUG 是否没有定义, 如果没有则以下代码是生效的,否则不生效 #ifndef DEBUG printf("__%d__%s__%s__\n" , __LINE__ , __FUNCTION__ , __FILE__ ); #endif
- 形式2:判断某一个宏是否被定义了或是否没有被定义,据此决定其所包含的代码段是否要编译
-
- 注意:#if形式条件编译需要有值宏
- 如果宏没有定义而且条件编译的判断语句是使用 形式1 (判断值) ,则表达式默认不成立(默认为假)。
#define A 0
#define B 1
#define C 2
#if A
... // 如果 MACRO 为真,那么该段代码将被编译,否则被丢弃
#endif
// 二路分支
#if A
...
#elif B
...
#endif
// 多路分支
#if A
...
#elif B
...
#elif C
...
#endif
实例:
#if DEBUG // 判断宏 DEBUG 是否为真, 如果是则编译 以下代码块
printf("__%d__%s__%s__\n" , __LINE__ , __FUNCTION__ , __FILE__ );
#elif TEST // 如果宏 DEBUG 为假, 则判断 宏 TEST 是否为真 ,如果为真则生效以下代码块
printf("__%d__%s__%s__\n" , __LINE__ , __FUNCTION__ , __FILE__ );
#else // 如果 DEBUG 为假 且 宏 TEST 是为假 ,则生效以下代码块
printf("__%d__%s__%s__\n" , __LINE__ , __FUNCTION__ , __FILE__ );
#endif // 条件判断的结束
- 总结:
- #ifdef 或 #ifndef 此种形式,判定的是宏是否已被定义,这不要求宏有值(有值无值都可以)。
- #if 、#elif 这些形式,判定的是宏的值是否为真,这要求宏必须有值。
- 只要出现了 #if #ifdef #ifndef 就必须在最后加上 #endif 表示结束判断
条件编译的使用场景
控制调试语句:在程序中,用条件编译将调试语句包裹起来,通过gcc编译选项随意控制调试代码的启停状态。例如:
gcc example.c -o example -DDEBUG // 通过命令行编译的时候定义宏 DEBUG 如果没有赋值默认为真
gcc example.c -o example -DDEBUG=0 -DTEST=1 // 在定义宏 DEBUG的时候赋值为 0 定义 宏TEST 的时候赋值为 1
以上语句中,-D意味着 Define,DEBUG 是程序中用来控制调试语句的一个宏,如此一来就可以在完全不需要修改源代码的情况下,通过外部编译指令选项非常方便地控制调试信息的启停。