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++ 语言中,“宏” 分为有参数和无参数两种。

一、无参宏定义

  1. 无参宏定义的一般形式为:#define 标识符 字符串

  2. 其中的 “#” 表示这是一条预处理命令。凡是以 “#” 开头的均为预处理命令。“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)

还要说明的是:

  1. 宏定义是用宏名来表示一个字符串,在宏展开时又以该字符串取代宏名,这只是一种简单的代换,字符串中可以含任何字符,可以是常数,也可以是表达式,预处理程序对它不作任何检查。如有错误,只能在编译已被宏展开后的源程序时发现。

  2. 宏定义不是说明或语句,在行末不必加分号,如加上分号则连分号也一起置换。

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 中的大数。

转自(https://www.cnblogs.com/challenger-vip/p/3386819.html)

posted @ 2024-05-13 10:16  viewoverlook  阅读(2)  评论(0编辑  收藏  举报