C语言中条件编译详解

前言
    —般情况下,c语言源程序中的每一行代码.都要参加编译。但有时候出于对程序代码优化的考虑,希望只对其中一部分内容进行编译。此时就需要在程序中加上条件,让编译器只对满足条件的代码进行编译,将不满足条件的代码舍弃,这就是条件编译,相关的预编译指令如下:
#define
定义一个预处理宏
#undef
取消宏的定义
#if
编译预处理中的条件命令,相当于c语法中的if语句
#ifdef
判断某个宏是否被定义,若已定义,执行随后的语句
#ifndef
与#ifdef相反,判断某个宏是否未被定义
#elif
若#if,  #ifdef, #ifndef或前面的#elif条件不满足,则执行#elif之后的语句,相当于c语法中的else-if
else
与#if,  #ifdef, #ifndef对应,若这些条件不满足,则执行#else之后的语句,相当于c语法中的else
endif
#if, #ifdef, #ifndef这些条件命令的结束标志.
defined
与#if,  #elif配合使用,判断某个宏是否被定义
条件编译功能
    预处理程序提供了条件编译的功能。条件编译允许只编译源文件中满足条件的程序 段,使生成的目标程序较短,从而减少了内存的开销,并提高程序的效率,可以按不同的 条件去编译不同的程序部分,因而产生不同的目标代码文件,提高了程序的可移植性和灵活性。
应用举例
#undef
    可以取消宏定义,与#define对应。
#define
    #define命令定义一个宏,按照是否带参数通常分为对象宏、函数宏两种。
对象宏
    不带参数的宏被称为对象宏(objectlike macro)。对象宏多用于定义常量、通用标识。例如:
            // 常量定义#define max_length 100// 通用标识,日志输出宏#define slog printf// 预编译宏#define _debug  
函数宏
    带参数的宏。利用宏可以提高代码的运行效率:子程序的调用需要压栈出栈,这一过程如果过于频繁会耗费掉大量的cpu运算资源。所以一些代码量小但运行频繁的代码如果采用带参数宏来实现会提高代码的运行效率。但多数c++程序不推荐使用函数宏,调试上有一定难度,可考虑使用c++的inline代替之。例如:
        // 最小值函数#define min(a,b) ((a)>(b)? (a):(b))// 安全释放内存函数#define safe_delete(p) {if(null!=p){delete p; p = null;}}  
    defined用来测试某个宏是否被定义。defined(name): 若宏被定义,则返回1,否则返回0。可用于在一条判断语句中声明多个判别条件,例如:
  #if defined(vax) && defined(unix) && !defined(debug)   
#ifdef、#ifndef、#else、#endif
#ifdef用于判断某个宏是否定义,和#ifndef功能正好相反,二者仅支持判断单个宏是否已经定义。
              #ifdef abc// ... codes while definded abc#elif (version > 2)// ... codes while code_version > 2#else// ... remained cases#endif //  
        #ifndef abcd_h#define abcd_h// ... some declaration codes#endif // #ifndef abcd_h  
 #if、#elif、#else、#endif
#if可支持同时判断多个宏的存在,与常量表达式配合使用。常用格式如下:
                    #if 常量表达式1// ... some codes#elif 常量表达式2// ... other codes#elif 常量表达式3// ......#else// ... statement#endif     常量表达式可以是包含宏、算术运算、逻辑运算等等的合法c常量表达式,如果常量表达式为一个未定义的宏,那么它的值被视为0。
特别注意
#if 和 #ifdef的区别     在判断某个宏是否被定义时,应当避免使用#if,因为该宏的值可能就是被定义为0。而应当使用#ifdef或#ifndef。看两段段代码,哪段代码会被编译进去,强调下是编译进去,不是执行。
              #define xxx 0// 第一段条件编译#ifdef xxx 逻辑1#else 逻辑2#endif  
              #define xxx 0// 第二段条件编译#if xxx 逻辑1#else 逻辑2#endif  
第一段条件编译:逻辑1会被编译进去
第二段条件编译:逻辑2会被编译进去
区别:
#if既关心宏是否定义,又关心宏的逻辑的真假。
#ifdef(#if defined())、#ifndef(#if !defined())仅关心宏是否被定义,不关心宏的逻辑真假。
#if 0  或  #if 1注释
    当屏蔽掉大块代码时,使用#if 0比使用/**/要好。(因为用/**/做大段的注释时,需要防止被注释掉的代码段中有嵌套的/**/,一旦出现/**/嵌套/**/的情况,会导致你注释掉的代码区域并不是你想要的区域范围) 。
1)常见的一种,如有一段不想要的代码,可以直接用#if 0 ... #endif形式来注释,效果等同于/**/
    #if 0    ...code...#endif  
    2)选择结构的条件编译。(如果常量为真【非0,随便什么数字,只要不是0】,就执行程序段1,否则执行程序段2。)
          #if constant    ...code 1...#else    ...code 2...#endif  
    3)嵌套情况。(如果常量a为真【非0,随便什么数字,只要不是0】,就执行程序段1。当常量a为0且常量b为真时,执行程序段2;当常量a为0且常量b为0时,执行程序段3)
                  #if constant a  ...code1...#else        #if constant b        ...code 2...        #else        ...code 3...    #endif#endif  
总结
    有些程序在调试、兼容性、平台移植等情况下可能想要通过简单地设置一些参数就生成一个不同的软件,这当然可以通过变量设置,把所有可能用到的代码都写进去,在初始化时配置,但在不同的情况下可能只用到一部分代码,就没必要把所有的代码都写进去,就可以用条件编译,条件编译是为了让程序在各种不同的软硬件环境下都以运行,提高其可移植性。
原文标题:c/c++中条件编译#if、#elif、#ifdef等灵活秒用
文章出处:【微信公众号:嵌入式arm】欢迎添加关注!文章转载请注明出处。

中国广电进入四大运营商时代 正式开放移动通信业务
德州仪器裁撤中国区MCU团队 原MCU产品线迁往印度
“飞入寻常百姓家”前,智能驾驶还缺点什么?
FREETALK Everyman 高清网络摄像头现面向 S
22年零跑C11充电时空气开关跳闸
C语言中条件编译详解
基于NiosII的高精度数控直流稳压电源设计
美光表示下半年旗舰手机内存有望提升到16GB
一文读懂AW516x zigbee如何自组网功能
移动电源电芯优缺点分析
确保服务器的安全需要一些有效的措施
美的低糖电饭煲具有哪些优势
什么是LTPS
插电式混合动力和非插电混合动力区别
OPPO、vivo、小米宣布5G手机最快明年和消费者见面
乌克兰推出ACS-3(Raybird-3)无人机系统
中国科学院稀土研究院-PLC采集项目案例
ADI 音频处理专家与 Damson Global 世界级音频系统设计师之间的故事
如何为现有工厂提供IIoT解决方案
这家公司自主研发的新型低成本加速度传感器迎来“爆发式”需求