第九章预处理
对于宏定义说明以下几点: 对于宏定义说明以下几点:
1. 宏定义是用宏名来表示一个字符串,在宏展 宏定义是用宏名来表示一个字符串, 开时又以该字符串取代宏名, 开时又以该字符串取代宏名,这只是一种简单 的代换,字符串中可以含任何字符, 的代换,字符串中可以含任何字符,可以是常 也可以是表达式, 数,也可以是表达式,预处理程序对它不作任 何检查。如有错误, 何检查。如有错误,只能在编译已被宏展开后 的源程序时发现。 的源程序时发现。 2. 宏定义不是说明或语句,在行末不必加分号, 宏定义不是说明或语句,在行末不必加分号, 如加上分号则连分号也一起置换。 如加上分号则连分号也一起置换。
预处理功能
9、1 、 宏定义
9、2 文件包含 、 9、3 条件编译 、
9.1
宏定义
在C语言源程序中允许用一个标识符来表 示一个字符串,称为“ 被定义为“ 示一个字符串,称为“宏”。被定义为“宏” 的标识符称为“宏名” 在编译预处理时, 的标识符称为“宏名”。在编译预处理时,对 程序中所有出现的“宏名” 程序中所有出现的“宏名”,都用宏定义中的 字符串去代换, 这称为“宏代换” 字符串去代换, 这称为“宏代换”或 宏展开” “宏展开”。 宏定义是由源程序中的宏定义命令完成的。 宏定义是由源程序中的宏定义命令完成的。 宏代换是由预处理程序自动完成的。在C语言 宏代换是由预处理程序自动完成的。 分为有参数和无参数两种。 中,“宏”分为有参数和无参数两种。
无参宏定义
1、格式:#define 标识符 字符串 其中宏名为C语言的标识符,一般用大写的字 母表示。“字符串”可以是常数、表达式、格 式串等。在前面介绍过的符号常量的定义就是 一种无参宏定义。此外,常对程序中反复使用 的表达式进行宏定义。例如: # define M (y*y+3*y) 2、功能:用于将一个指定的标识符替换为一个字 符串。
第九章 编译预处理
在源程序中放在函数之外, 而且一般都放在源文 件的前面,它们称为预处理部分。 所谓预处理是指在进行编译的第一遍扫描(词法扫 描和语法分析)之前所作的工作。预处理是C语言的一 个重要功能, 它由预处理程序负责完成。当对一个源 文件进行编译时, 系统将自动引用预处理程序对源程 序中的预处理部分作处理,处理完毕自动进入对源程 序的编译。 预处理命令不是C语言本身的组成部分,不能直接对 它们进行编译。 注意:预处理命令以“#”开头;
7. 可用宏定义表示数据类型,使书写方便。 可用宏定义表示数据类型,使书写方便。 例如: 在程序中可用STU作变 例如: #define STU struct stu在程序中可用 在程序中可用 作变 量说明: 量说明: STU body[5],*p; 应注意用宏定义表示数据类型和用typedef定义数据 应注意用宏定义表示数据类型和用 定义数据 说明符的区别。宏定义只是简单的字符串代换, 说明符的区别。宏定义只是简单的字符串代换,是在预 处理完成的, 是在编译时处理的, 处理完成的,而typedef是在编译时处理的,它不是作简 是在编译时处理的 单的代换,而是对类型说明符重新命名。 单的代换,而是对类型说明符重新命名。被命名的标识 符具有类型定义说明的功能。请看下面的例子: 符具有类型定义说明的功能。请看下面的例子: #define PIN1 int* 与 typedef (int*) PIN2;从形式上看这两者 从形式上看这两者 相似, 但在实际使用中却不相同。下面用PIN1,PIN2说 相似, 但在实际使用中却不相同。下面用 , 说 明变量时就可以看出它们的区别: 明变量时就可以看出它们的区别: PIN1 a,b;在宏代换后变成 int *a,b;表示 是指向 表示a是指向 在宏代换后变成 表示 整型的指针变量, 是整型变量。 整型的指针变量,而b是整型变量。然而:PIN2 a,b;表 是整型变量 然而: 表 示a,b都是指向整型的指针变量。因为PIN2是一个类型说 都是指向整型的指针变量。因为 是一个类型说 都是指向整型的指针变量 明符。由这个例子可见,宏定义虽然也可表示数据类型, 明符。由这个例子可见,宏定义虽然也可表示数据类型, 但毕竟是作字符代换。在使用时要分外小心,以避出错。 但毕竟是作字符代换。在使用时要分外小心,以避出错。
4、带参数的宏替换与函数有些相似,但有不同: 、带参数的宏替换与函数有些相似,但有不同: 宏替换产生的程序代码比使用函数时长。 1)、宏替换产生的程序代码比使用函数时长。 宏替换不需要进行参数传递、保护现场、 2)、宏替换不需要进行参数传递、保护现场、等操 作其执行速度比函数快。 作其执行速度比函数快。 3)、宏替换不存在返回值类型和参数类型的问题。 宏替换不存在返回值类型和参数类型的问题。 4)、使用宏时应注意: ) 使用宏时应注意: 定义宏时,最好对参数和宏体用括号括起来。 A、定义宏时,最好对参数和宏体用括号括起来。 如:#define square(n) n*n s=aquare(n+1 调用时 s=aquare(n+1) 则变成了s=a+1*a+1与预期效果不同。 则变成了s=a+1*a+1与预期效果不同。 s=a+ 定义带参数的宏时, B、定义带参数的宏时,宏名和左括号之间不能有空 否则被认为是无参数的宏. 格,否则被认为是无参数的宏. C、对宏的定义 只是进行字符的替换 不进行语法检 只是进行字符的替换,不进行语法检 、对宏的定义,只是进行字符的替换 查.
3. 宏定义必须写在函数之外,其作用域为宏定 义命令起到源程序结 束。如要终止其作用域可 使用# undef命令,例如: # define PI 3.14159 main() { …… } # undef PIPI的作用域 f1() 表示PI只在main函数中有效,在f1中无效。
4. 宏名在源程序中若用引号括起来,则预处理程 宏名在源程序中若用引号括起来, 序不对其作宏代换。 序不对其作宏代换。intf("OK"); printf("\n"); } 上例中定义宏名OK表示 表示100,但在 上例中定义宏名 表示 ,但在printf语 语 句中OK被引号括起来,因此不作宏代换。程序的 被引号括起来, 句中 被引号括起来 因此不作宏代换。 运行结果为: 这表示把 这表示把“ 当字符串处理。 运行结果为:OK这表示把“OK”当字符串处理。 当字符串处理
6. 宏定义也可用来定义多个语句,在宏调用时,把 宏定义也可用来定义多个语句,在宏调用时, 这些语句又代换到源程序内。看下面的例子。 这些语句又代换到源程序内。看下面的例子。 #define SSSV(s1,s2,s3,v) s1=l*w;s2=l*h;s3=w*h;v=w*l*h; main() { int l=3,w=4,h=5,sa,sb,sc,vv; SSSV(sa,sb,sc,vv); printf("sa=%d\nsb=%d\nsc=%d\nvv=%d\n",sa,s b,sc,vv); } 程序第一行为宏定义,用宏名SSSV表示 个赋值语 表示4个赋值语 程序第一行为宏定义,用宏名 表示 个形参分别为4个赋值符左部的变量 个赋值符左部的变量。 句,4 个形参分别为 个赋值符左部的变量。在宏调 用时, 个语句展开并用实参代替形参。 用时,把4 个语句展开并用实参代替形参。使计算结 果送入实参之中。 果送入实参之中。
5. 宏定义允许嵌套,在宏定义的字符串中 宏定义允许嵌套, 可以使用已经定义的宏名。 可以使用已经定义的宏名。在宏展开时由 预处理程序层层代换。例如: 预处理程序层层代换。例如: #define PI 3.1415926 #define S PI*y*y /* PI是已定义的 PI是已定义的 宏名*/对语句 对语句: 宏名 对语句: printf("%f",s);在宏代换后 在宏代换后 变为: 变为: printf("%f",3.1415926*y*y); 6. 习惯上宏名用大写字母表示,以便于与 习惯上宏名用大写字母表示, 变量区别。但也允许用小写字母。 变量区别。但也允许用小写字母。
4. 在宏定义中,字符串内的形参通常要用括号括 在宏定义中, 起来以避免出错。 起来以避免出错。 #define SQ(y) y*y 代换成: 代换成:sq=a+1*a+1; 注意:对于宏定义不仅应在参数两侧加括号, 注意:对于宏定义不仅应在参数两侧加括号, 也 应在整个字符串外加括号。 应在整个字符串外加括号。 5. 带参的宏和带参函数很相似,本质上不同, 带参的宏和带参函数很相似,本质上不同,
对于带参的宏定义说明: 对于带参的宏定义说明:
1. 带参宏定义中,宏名和形参表之间不能有空格出现。 带参宏定义中,宏名和形参表之间不能有空格出现。 例如把: 写为: 例如把: #define MAX(a,b) (a>b)?a:b 写为: #define MAX (a,b) (a>b)?a:b 将被认为是无参宏定义, 将被认为是无参宏定义, 宏名MAX代表字符串 (a,b) (a>b)?a:b。宏展开时,宏调用 宏名 代表字符串 。宏展开时, 语句: 将变为: 语句: max=MAX(x,y);将变为: 将变为 max=(a,b)(a>b)?a:b(x,y);这显然是错误的。 这显然是错误的。 这显然是错误的 2. 在带参宏定义中,形式参数不分配内存单元,因此不必作 在带参宏定义中,形式参数不分配内存单元, 类型定义。而宏调用中的实参有具体的值。 类型定义。而宏调用中的实参有具体的值。要用它们去代 换形参,因此必须作类型说明。 换形参,因此必须作类型说明。这是与函数中的情况不同 在函数中,形参和实参是两个不同的量, 的。在函数中,形参和实参是两个不同的量,各有自己的 作用域,调用时要把实参值赋予形参,进行“值传递” 作用域,调用时要把实参值赋予形参,进行“值传递”。 而在带参宏中,只是符号代换,不存在值传递的问题。 而在带参宏中,只是符号代换,不存在值传递的问题。
二、带参数的宏替换
1、定义:#define 宏名(形式参数表) 宏体 、定义: 宏名(形式参数表) 2、引用:宏名(实际参数表) 、引用:宏名(实际参数表) 3、对带参数的宏,不仅对宏名进行替换,而且对参数 、对带参数的宏,不仅对宏名进行替换, 进行替换。 进行替换。 例如: <stdio. 例如:#include <stdio.h> #define PI 3.14 #define circuit(r) 2*PI*r #define area(r) PI*r*r main() { float ra,c,a; ra,c,a; scanf(“% ,&ra) ,&ra); scanf( %f”,&ra); c=circuit(ra); c=circuit(ra); a=area(ra); a=area(ra); printf(“ra= f,c=%f,a=% ,ra,c,a) ra=% ,ra,c,a); printf( ra=%f,c=%f,a=%f”,ra,c,a);}