所谓预处理是指在进行编译的第一遍(词法扫描和语法分析)之前所做的工作。预处理是C语言的一个重要功能,它由预处理程序负责完成。当对一个原文件进行编译时,系统将自动引用预处理程序对源程序中的预处理部分进行处理,处理完毕自动进入对源程序的编译。
第一部分 宏定义
在C语言源程序中允许用一个标识符来表示一个字符串,成为“宏”。被定义为“宏”的标识符成为“宏名”。在编译处理命令时,对程序中所出现的“宏名”,都是宏定义中的字符串去代替,这称为“宏代换”或“宏展开”。
宏定义是由源程序中的宏定义命令完成的。宏代换是由预处理程序自动完成的。
在C语言中,“宏”分为有参数和无参数两种。
1.1 无参宏定义
无参宏的宏名不带参数。其定义的一般形式为:
#define 标识符 字符串
其中“#”表示这是一条预处理命令。凡是以“#”开头的均为预处理命令。“define“为宏定义命令。“标识符”为所定义的宏名。“字符串”可以是常量、表达式或格式串等。
在前面介绍过的符号常量的定义就是一种无参宏定义。此外,对程序中反复使用的表达式也可进行宏定义。如:#define M (y*y+3*y)
它的作用是指定标识符M来代替表达式(y*y+3*y)。在编写源程序时,所有的(y*y+3*y)都可由M代替,而对源程序进行编译时,将先由预处理程序进行宏代换,即用(y*y+3*y)表达式去置换所有的宏名M,然后再进行编译。
例1:红的定义和替换
#include<stdio.h>
#define M (y*y+3*y)
main(){
int s,y;
printf("input a number:/n");
scanf("%d",&y);
s=3*M+4*M+5*M;
printf("s=%d/n",s);
}
首先定义了宏,定义M来替换表达式(y*y+3*y),在s=3*M+4*M+5*M;中进行宏调用。在预处理时经宏展开后该语句变为:s=3*
(y*y+3*y)+4*(y*y+3*y)+5*(y*y+3*y);但是要注意在宏定义中表达式(y*y+3*y)两边的括号不能少。否则会产生错误。
对于宏定义有几点说明:
(1)宏定义是用宏名来表示一个字符串,在宏展开时又以该字符串取代宏名,这只是一种简单的代换,字符串中可以包含任何字符字符,可以是常量,也可以是表达式,预处理程序对他不做任何检查。如有错误,只能在编译已被展开后的源程序时发现。
(2)宏定义不是类型声明或语句,在行末不必加分号,如加分号则连分号也一起置换。
(3)宏定义必须写在函数之外,其作用于从宏定义命令起到源程序结束。如要终止起作用域也是用#undef命令。
(4)宏名在源程序中若用引号括起来,则预处理程序不对其进行宏代换。
(5)宏定义允许嵌套,在宏定义的字符串中可以使用已经定义的宏名。在宏展开时由预处理程序层层代换。
(6)习惯上宏名用大写字符表示,以便与变量区分。但也允许用小写字母。
(7)可用宏定义表示数据类型,使书写方便。
如:
#define STU struct stu
在程序中可用STU做变量说明:STU body[5],*p;
#define INTEGER int
在程序中就可用INTEGER进行整形变量说明:INTEGER a,b;
应注意用宏定义表示数据类型和typedef定义数据说明符的区别。宏定义只是简单的字符串替换,是在预处理阶段完成的,而typedef是在编译阶段进行处理的,他不是做简单的代换,而是对类型说明符重新命名。被命名的标识符具有类型定义说明的功能。
请看下面的例子:
#define PIN1 int *
typedef (int *)PIN2;
从形式上看这两者相似,但在实际使用中却不相同。下面用PIN1,PIN2声明变量就可以看出他们的区别:
PIN1 a,b;在宏替换后变成: int * a,b;表示a是指向整形的指针变量,而b是整型变量。然而:PIN2 a,b;表示a,b都是指向整形的指针变量。因为PIN2是一个类型说明符。由此例可知,宏定义虽然也可表示数据类型,但毕竟是做字符替换。在使用时要格外小心,避免出错。
例3:对“输入格式”做宏定义
#include<stdio.h>
#define P printf
#define D "%d,"
#define F "%f/n"
main(){
int a=5,c=8,e=11;
float b=3.8,d=8.7,f=21.08;
P(D F,a,b);
P(D F,c,d);
P(D F,e,f);
}
1.2 带参宏定义
C语言允许宏带有参数。在宏定义中的参数成为形式参数,在宏调用中的参数成为实际参数。对于带参数的宏,在调用中,不仅要进行宏展开,而且要用实参去替换形参。
带参宏定义的一般形式为:#define 宏名(形参表) 字符串
在字符串中含有各个形参。带参宏调用的一般格式为:宏名(实参表);
如:#define M(y) y*y+3*y ... k=M(5); ...
例4:带参数的宏
#include<stdio.h>
#define MAX(a,b) (a>b)?a:b
main(){
int x,y,max;
printf("input two numbers:/n");
scanf("%d%d",&x,&y);
max=MAX(x,y);
printf("max=%d/n",max);
}
对带参的宏定义有以下几点说明:
(1)带参宏定义中,宏名和形参表之间不能有空格。
(2)在带参宏定义中,形式参数不分配内存空间,因此不必做类型定义。而宏调用中的实参有具体的值。要用它们去替换形参,因此必须做类型说明。这是与函数中的情况不同的。在函数中,形参和实参是两个不同的量,各有自己的作用域,调用时要把实参值赋值予形参,进行“值传递”。而在带参宏中,只进行符号替换,不存在值传递的问题。
(3)在宏定义中的形参是标识符,而宏调用中的实参可以是表达式。
(4)宏定义中,字符串内的形参通常把实参表达式的值用括号括起来以避免出错。
(5)带参的宏和带参的函数很相似,但本质上是不同的,除上面已谈到的几点外,把同一表达式用函数处理与用宏处理后,两者的结果可能是不同的。
例5:带参的函数
#include<stdio.h>
main(){
int i=1;
while(i<5)
printf("%d ",SQ(i++));
}
SQ(int y){
return (y*y);
}
例6:带参的宏
#include<stdio.h>
#define SQ(y) (y*y)
main(){
int i=1;
while(i<=5)
printf("%d ",SQ(i++));
}