原创 条件编译/条件宏定义

2009-11-16 15:48 3655 3 3 分类: 软件与OS

                                                                                          ---xiaoxiaopig       2009/11/16


条件性编译


一般情况下,源程序中所有的行都参加编译。但是有时希望对其中一部分内容只在满足一定条件下才进行编译,即对一部分内容指定编译条件,这就是“条件编译”。(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
重要解释:若宏NNN为True则只留下statement1编译;若NNN为False且MMM为True则只编译statement2;若NNN和MMM都为False则编译statement3。


#if是在编译前进行抉择的,而一般的if指令是在程序运行时才做抉择的,因此#if可以提升程序的执行速度,这是两者的重要区别。另外,#if指令还可协助查错。


条件性定义


#ifndef


  #ifndef x

  #define x

  ...

  #endif

  这是宏定义的一种,它可以根据是否已经定义了一个变量来进行分支选择,一般用于调试等等

  #ifndef x

  //先测试x是否被定义过

  #define x

  //如果没有定义下面就定义x并执行下面的语句

  ...

  #endif

  //如果已经定义了则执行#endif后面的语句

  条件指示符#ifndef检查预编译常量在前面是否已经被定义。如果在前面没有被定义,则条件指示符的值为真,于是从#ifndef到#endif之间的所有语句都被包含进来进行处理。相反,如果#ifndef指示符的值为假,则它与#endif指示符之间的行将被忽略。条件指示符#ifndef 的最主要目的是防止头文件的重复包含和编译。

  补充一些内容:头文件中的#ifndef

  千万不要忽略了头件的中的#ifndef,这是一个很关键的东西。比如你有两个C文件,这两个C文件都include了同一个头文件。而编译时,这两个C文件要一同编译成一个可运行文件,于是问题来了,大量的声明冲突。

  还是把头文件的内容都放在#ifndef和#endif中吧。不管你的头文件会不会被多个文件引用,你都要加上这个。一般格式是这样的:

  #ifndef <标识>

  #define <标识>

  ......

  ......

  #endif

  <标识>在理论上来说可以是自由命名的,但每个头文件的这个“标识”都应该是唯一的。标识的命名规则一般是头文件名全大写,前后加下划线,并把文件名中的“.”也变成下划线,如:stdio.h

  #ifndef _STDIO_H_

  #define _STDIO_H_

  ......

  #endif

重要补充说明:


#ifdef 宏   //若已定义了此宏,则留下#ifdef与#endif间的指令;否则删除之。


#ifndef 宏 //若未定义过此宏,则留下#ifndef与#endif间的指令;否则删除之。


#endif //定义#ifdef及#ifndef的范围。


#undef 宏 //与#defined相反的动作---解除定义。


#else  //可构成#ifdef~#else~#endif结构或#ifndef~#else~#endif结构。


 


#ifdef与#if的区别


#if 宏:此宏必须已定义,依宏所代表的值来做判断;


#ifdef 宏:此宏不一定已定义,依此宏是否已定义来判断。

PARTNER CONTENT

文章评论0条评论)

登录后参与讨论
EE直播间
更多
我要评论
0
3
关闭 站长推荐上一条 /3 下一条