资源预览内容
第1页 / 共7页
第2页 / 共7页
第3页 / 共7页
第4页 / 共7页
第5页 / 共7页
第6页 / 共7页
第7页 / 共7页
亲,该文档总共7页全部预览完了,如果喜欢就下载吧!
资源描述
C 语言的条件编译#if, #elif, #else , #endif、#ifdef, #ifndef(2011-03-25 14:15:58) 转载标签: 宏都代码段条件指令c 语言it分类: C 语言编程 有些程序在调试、兼容性、平台移植等情况下可能想要通过简单地设置一些参数就生成一个不同的软件,这当然可以通过变量设置,把所有可能用到的代码都写进去,在初始化时配置,但在不同的情况下可能只用到一部分代码,就没必要把所有的代码都写进去,就可以用条件编译,通过预编译指令设置编译条件,在不同的需要时编译不同的代码。 (一)条件编译方法条件编译是通过预编译指令来实现的,主要方法有:1、#if, #elif, #else, #endif#if条件1代码段1#elif 条件 2代码段2.#elif 条件 n代码段n#else代码段 n+1#endif即可以设置不同的条件,在编译时编译不同的代码,预编译指令中的表达式与 C 语言本身的表达式基本一至如逻辑运算、算术运算、位运算等均可以在预编译指令中使用。之所以能够实现条件编译是因为预编译指令是在编译之前进行处理的,通过预编译进行宏替换、条件选择代码段,然后生成最后的待编译代码,最后进行编译。 #if 的一般含义是,如果#if 后面的常量表达式为 true,则编译它所控制的代码,如条件 1 成立时就代码段 1,条件 1 不成立再看条件 2 是否成立,如果条件 2 成立则编译代码段 2,否则再依次类推判断其它条件,如果条件 1N 都不成力则会编译最后的代码段 n+1.2、#ifdef, #else, #endif 或#ifndef, #else, #endif条件编译的另一种方法是用#ifdef 与#ifndef 命令,它们分别表示 “如果有定义”及“如果无定义”。有定义是指在编译此段代码时是否有某个宏通过 #define 指令定义的宏,#ifndef 指令指找不到通过 #define 定义的某宏,该宏可以是在当前文件此条指令的关面定义的,也可以是在其它文件中,但在此指令之前包含到该文件中的。#ifdef 的一般形式是:#ifdef macro_name代码段1#else代码段2#endif或#ifdef 的一般形式是:#ifndef macro_name代码段2#else代码段1#endif这两段代码的效果是完全一样的。3、通过宏函数 defined(macro_name )参数为宏名(无需加),如果该 macro_name 定义过则返回真,否则返回假,用该函数则可以写比较复杂的条件编译指令如#if defined(macro1) | (!defined(macro2) & defined(macro3).#else.#endif(二)条件编译技巧与示例 (1)#ifdef 和#defined()比较首先比较一下这两种方法,第一种方法只能判断一个宏,如果条件比较复杂实现起来比较烦锁,用后者就比较方便。如有两个宏 MACRO_1,MACRO_2只有两个宏都定义过才会编译代码段 A,分别实现如下:#ifdef MACRO_1#ifdef MACRO_2代码段A#endif#endif或者#if defined(MACRO_1) & defined(MACRO_2)#endif同样,要实现更复杂的条件用#ifdef 更麻烦,所以推荐使用后者,因为即使当前代码用的是简单的条件编译,以后在维护、升级时可能会增加,用后者可维护性较强。旧的编译器可能没有实现#defined()指令,C99 已经加为标准。要兼容老的编译器,还需用#ifdef 指令。2、#if 与 #ifdef 或#ifdefined ()比较比如自己写了一个 printf 函数,想通过一个宏 MY_PRINTF_EN 实现条件编译,用#if 可实现如下C 语言的条件编译。#define MY_PRINTF_EN 1#if MYS_PRINTF_EN = 1int printf(char* fmt, char* args, .).#endif如果宏 MY_PRINTF_EN 定义为 1 则编译这段代码,如果宏定义不为 1 或者没有定义该宏,则不编译这段代码。同样也可以通过#ifdef 或者#defined()实现,如#define MY_PRINTF_EN 1#if defined(MY_PRINTF_EN)int printf(char* fmt, char* args, .).#endif在这种情况下两种方法具有异曲同工之妙,但试想如果你为了节约代码写了两个 printf 函数,在不同情况下使用不同的 printf 函数,一个是精简版一个是全功能标准版,如:#define MY_PRINTF_SIMPLE#ifdef MY_PRINTF_SIMPLEvoid printf(*str) / 向终端简单地输出一个字符串.#endif#ifdef MY_PRINTF_STANDARDint printf(char* fmt, char* args, .).#endif同样可以用#if defined()实现#define MY_PRINTF_SIMPLE#if defined(MY_PRINTF_SIMPLE)void printf(*str) / 向终端简单地输出一个字符串.#elif defined(MY_PRINTF_STANDARD)int printf(char* fmt, char* args, .).#endif两种方法都可以实现,但可见后者更方便。但试想如果你有三个版本,用前者就更麻烦了,但方法相似,用后者就更方便,但仍需三个宏进行控制,你要住三个宏,改进一下就用#if 可以用一个宏直接控制 N 种情况如:#define MY_PRINTF_VERSION 1#if MY_PRINTF_VERSION = 1void printf(*str) / 向终端简单地输出一个字符串.#elif MY_PRINTF_VERSION = 2int printf(char* fmt, char* args, .).#elif MY_PRINTF_VERSION = 3int printf(unsigned char com_number, char* str).#else默认版本#endif这样,你只需修改一下数字就可以完成版本的选择了看来好像用#if 比较好了,试想如下情况:你写了一个配置文件叫做config.h 用来配置一些宏,通过这些宏来控制代码,如你在 config.h 的宏#define MY_PRINTF_EN 1来控制是否需要编译自己的 printf 函数,而在你的源代码文件 printf.c 中有如下指令i nclude config.h#if MY_PRINTF_EN = 1int printf(char* fmt, char* args, .).#endif但这样也会有一个问题,就是如果你忘了在 config.h 中添加宏MY_PRINTF_EN,那么自己写的 printf 函数也不会被编译,有些编译器会给出警告:MY_PRINTF_EN 未定义。如果你有两个版本的想有一个默认版本,可以在 printf.c 中这样实现#incldue config.h#if !defined(MY_PRINTF_VERSION)#define MY_PRINTF_VERSION 1#endif#if MY_PRINTF_VERSION = 1void printf(*str) / 向终端简单地输出一个字符串.#elif MY_PRINTF_VERSION = 2int printf(char* fmt, char* args, .).#elif MY_PRINTF_VERSION = 3int printf(unsigned char com_number, char* str).#endif这种情况下还得用到#ifdef 或#ifdefined (),你可以不用动主体的任何代码,只需要修改 printf.c 文件中 MY_RPINTF_VERSION 宏的数字就可以改变了,如果用前面那种方法还得拖动代码,在拖动中就有可能造成错误。再试想,如果软件升级了,或者有了大的改动,原来有三个版本,现在只剩下两个版本了,如#if MY_PRINTF_VERSION = 2int printf(char* fmt, char* args, .).#elif MY_PRINTF_VERSION = 3int printf(unsigned char com_number, char* str).#endif因为这些核心代码不想让使用这些代码的人关心,他们只需要修改 config.h文件,那就要在 printf.c 中实现兼容性。如果以前有人在 config.h 配置宏MY_PRINTF_VERSION 为 1,即有#define MY_PRINTF_VERSION 1而现在没有 1 版本了,要想兼容怎么办?那当然可以用更复杂的条件实现如:#if MY_PRINTF_VERSION = 2 | MY_PRINTF_VERSION = 1int printf(char* fmt, char* args, .).#elif MY_PRINTF_VERSION = 3int printf(unsigned char com_number, char* str).#endif不过还有另外一种方法,即使用#undef 命令#if MY_PRINTF_VERSION = 1#undef MY_PRINTF_VERSION#define MY_PRINTF_VERSION 2#endif#if MY_PRINTF_VERSION = 2int printf(char* fmt, char* args, .).#elif MY_PRINTF_VERSION = 3int printf(unsigned char com_number, char* str).#endif用#if 还有一个好处,如果你把宏名记错了,把 MY_PRINTF_EN 定义成了 MY_PRINT_EN,那么你用#ifdef MY_PRINTF_EN 或者#if defined(MY_PRINTF_EN)控制的代码就不能被编译,查起来又不好查,用 #if MY_PRINTF_EN =1 控制就很好查,因为你把 MY_PRINTF_EN 定义成MY_PRINT_EN,则 MY_PRINTF_EN 实际上没有定义,那么编译器会给出警告#if MY_PRINTF_EN = 1 中的 MY_PRINTF_EN 没有定义,但错就比较快。
收藏 下载该资源
网站客服QQ:2055934822
金锄头文库版权所有
经营许可证:蜀ICP备13022795号 | 川公网安备 51140202000112号