ifndef_ define_ endif 作用
#ifndef
它是 if not define 的简写,是宏定义的一种,实际上确切的说,这应该是预处理功能三种(宏定义、文件包含、条件编译)中的一种 ---- 条件编译。
在 c 语言中,对同一个变量或者函数进行多次声明是不会报错的。所以如果 h 文件里只是进行了声明工作,即使不使用# ifndef 宏定义,多个 c 文件包含同一个 h 文件也不会报错。
但是在 c++ 语言中,#ifdef 的作用域只是在单个文件中。所以如果 h 文件里定义了全局变量,即使采用 #ifdef 宏定义,多个 c 文件包含同一个 h 文件还是会出现全局变量重定义的错误。
使用 #ifndef 可以避免下面这种错误:如果在 h 文件中定义了全局变量,一个 c 文件包含同一个 h 文件多次,如果不加 #ifndef 宏定义,会出现变量重复定义的错误;如果加了 #ifndef,则不会出现这种错误。
示例:
#ifndef x //先测试x是否被宏定义过
#define x
程序段1blabla~ //如果x没有被宏定义过,定义x,并编译程序段 1
#endif
程序段2blabla~ //如果x已经定义过了则编译程序段2的语句,“忽视”程序段 1
条件指示符 #ifndef
的最主要目的是防止头文件的重复包含和编译。了解:条件编译当然也可以用条件语句来实现。 但是用条件语句将会对整个源程序进行编译,生成的目标程序程序很长,而采用条件编译,则根据条件只编译其中的程序段 1 或程序段 2,生成的目标程序较短。如果条件选择的程序段很长,采用条件编译的方法是十分必要的。
#ifndef
和 #endif
要一起使用,如果丢失 #endif
,可能会报错。总结一下:在 c 语言中,对同一个变量或者函数进行多次声明是不会报错的。所以如果 h 文件里只是进行了声明工作,即使不使用# ifndef
宏定义,一个 c 文件多次包含同一个 h 文件也不会报错。 使用 #ifndef
可以避免下面这种错误:如果在 h 文件中定义了全局变量,一个 c 文件包含同一个 h 文件多次,如果不加 #ifndef
宏定义,会出现变量重复定义的错误;如果加了 #ifndef
,则不会出现这种错.
#ifdef
与 ifndef 类似,ifdef 顾名思义,就是 if define,看例子
#ifdef x
程序1blabla~
#endif
翻译:如果宏定义了 x,则执行程序 1.
此外,还有其他形式,还是看例子好些:
#ifndef x
#define x
程序段 1
#else
程序段 2
#endif
当 x 没有由 #define
定义过,则编译 “程序段 1”,否则编译 “程序段 2”。
#if 表达式
程序段 1
#else
程序段 2
#endif
它的作用是 当 “表达式” 值为真时。编译程序段 1。否则则编译程序段 2。当没有程序段 2 时,直接是 #if---#endif
#define
在 C 或 C++ 语言源程序中允许用一个标识符来表示一个字符串,称为 “宏”。“define” 为宏定义命令。
被定义为 “宏” 的标识符称为“宏名”。在编译预处理时,对程序中所有出现的“宏名”,都用宏定义中的字符串去代换,这称为 “宏代换” 或“宏展开”。宏定义是由源程序中的宏定义命令完成的。宏代换是由预处理程序自动完成的。
优点:
(1) 方便程序的修改。这个就不多说了。
(2) 提高程序的运行效率。使用带参数的宏定义可完成函数调用的功能,又能减少系统开销,提高运行效率。正如 C 语言中所讲,函数的使用可以使程序更加模块化,便于组织,而且可重复利用,但在发生函数调用时,需要保留调用函数的现场,以便子函数执行结束后能返回继续执行,同样在子函数执行完后要恢复调用函数的现场,这都需要一定的时间,如果子函数执行的操作比较多,这种转换时间开销可以忽略,但如果子函数完成的功能比较少,甚至于只完成一点操作,如一个乘法语句的操作,则这部分转换开销就相对较大了,但使用带参数的宏定义就不会出现这个问 题,因为它是在预处理阶段即进行了宏展开,在执行时不需要转换,即在当地执行。宏定义可完成简单的操作,但复杂的操作还是要由函数调用来完成,而且宏定义所占用的目标代码空间相对较大。所以在使用时要依据具体情况来决定是否使用宏定义。
在 C 或 C++ 语言中,“宏” 分为有参数和无参数两种。
一、无参宏定义
-
无参宏定义的一般形式为:#define 标识符 字符串
-
其中的 “#” 表示这是一条预处理命令。凡是以 “#” 开头的均为预处理命令。“define” 为宏定义命令。“标识符” 为所定义的宏名。“字符串” 可以是常数、表达式、格式串等。
1 #include <stdio.h>
2 #define M ( a+b )
3 int main( int argc, char * argv[] )
4 {
5 int s, a, b;
6 printf( "input number a& b: " );
7 scanf( "%d%d", &a, &b );
8 s = M*M;
9 printf( "s=%d\n" ,s );
10 }
上例程序中首先进行宏定义,定义 M 来替代表达式 (a+b), 在 s= M * M 中作了宏调用。在预处理时经宏展开后该语句变为: S=(a+b)(a+b) 但要注意的是,在宏定义中表达式 (a+b) 两边的括号不能少。否则会发生错误。 如当作以下定义后:#define M (a)+(b) 在宏展开时将得到下述语句:S= (a)+(b)(a)+(b)
还要说明的是:
-
宏定义是用宏名来表示一个字符串,在宏展开时又以该字符串取代宏名,这只是一种简单的代换,字符串中可以含任何字符,可以是常数,也可以是表达式,预处理程序对它不作任何检查。如有错误,只能在编译已被宏展开后的源程序时发现。
-
宏定义不是说明或语句,在行末不必加分号,如加上分号则连分号也一起置换。
3.. 宏定义必须写在函数之外,其作用域为宏定义命令起到源程序结束。如要终止其作用域可使用 #undef 命令
二、带参宏定义
c 语言允许宏带有参数。在宏定义中的参数称为形式参数,在宏调用中的参数称为实际参数。对带参数的宏,在调用中,不仅要宏展开,而且要用实参去代换形参。
带参宏定义的一般形式为: #define 宏名 (形参表) 字符串
例:
#define M(y) ((y)*(y)+3*(y)) /*宏定义*/
k = M(5); /*宏调用*/
#include <stdio.h>
#define MAX( a, b ) ((a>b)?(a):(b))
int main( int argc, char * argv[] ) {
int x, y, max;
printf( "input two numbers: " );
scanf( "%d%d", &x, &y );
max = MAX( x, y );
printf( "max=%d\n", max );
return 0;
}
上例程序的第一行进行带参宏定义,用宏名 MAX 表示条件表达式 (a>b)?a:b ,形参 a,b 均出现在条件表达式中。程序中 max=MAX(x,y) 为宏调用,实参 x,y,将代换形参 a,b。宏展开后该语句为: max=(x>y)?x:y; 用于计算 x,y 中的大数。