1、第六讲编译预处理第6讲 编译预处理6.1 编译预处理的根概念和特点预处理是指在进行编译的第一遍扫描(词法扫描和语法分析)之前所作的工作。6.1.2 编译预处理的特点预处理命令均以“#”号开头,在它前面不能出现空格以外的其他字符。每一行命令独占一行。命令不以“;”为结束符,它不是C语句。预处理程序控制行的作用范围仅限于说明它们的那个文件。语言提供的预处理功能有:宏定义、文件包含、条件编译等。合理地使用预处理功能编写的程序便于阅读、修改、移植和调试,也有利于模块化程序设计。6.2 宏定义 宏在语言源程序中允许用一个标识符来表示一个字符串,称为“宏”。宏名被定义为“宏”的标识符称为“宏名”。宏代换(
2、宏替换或宏展开)在编译预处理时,对程序中所有出现的“宏名”,都用宏定义中的字符串去代换,这称为“宏代换”或“宏展开”。6.2.1 不带参的宏定义其定义的一般形式为: #define 标识符 字符串在前面介绍过的符号常量的定义就是一种无参宏定义。 #define PI 3.14.5926常对程序中反复使用的表达式进行宏定义。例如: #define M (y*y+3*y)用标识符M来代替表达式(y*y+3*y)。在编写源程序时,所有的(y*y+3*y)都可由M代替。【例】#include#define M (y*y+3*y)void main() int s,y; printf(input a n
3、umber: ); scanf(%d,&y); s=3*M+4*M+5*M; printf(s=%dn,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)两边的括号不能少。否则会发生错误。如当作以下定义后: #difine M y*y+3*y在宏展开时将得到下述语句: s=3*y*y+3*y+4*y*y+3*y+5*y*y+3*y;这相当于: 3y2+3y+4y2+3y+5y2+3y;
4、显然与原题意要求不符。计算结果当然是错误的。【例】#include#define OK 100void main() printf(OK); printf(n);运行程序输出:OK程序的运行结果为:OK这表示把“OK”当字符串处理。6.2.2 带参宏定义语言允许宏带有参数。在宏定义中的参数称为形式参数,在宏调用中的参数称为实际参数。对带参数的宏,在调用中,不仅要宏展开,而且要用实参去代换形参。带参宏定义的一般形式为: #define 宏名(形参表) 字符串在字符串中含有各个形参。带参宏调用的一般形式为: 宏名(实参表); 例如: #define M(y) y*y+3*y /*宏定义*/ k=M
5、(5); /*宏调用*/ 在宏调用时,用实参5去代替形参y,经预处理宏展开后的语句为: k=5*5+3*5【例】#include#define MAX(a,b) (ab)?a:bvoid main() int x,y,max; printf(input two numbers: ); scanf(%d%d,&x,&y); max=MAX(x,y); printf(max=%dn,max);宏名MAX表示条件表达式(ab)?a:b,形参a,b均出现在条件表达式中。语句max=MAX(x,y)为宏调用,实参x,y,将代换形参a,b。宏展开后该语句为: max=(xy)?x:y;用于计算x,y中的大
6、数。说明:1)带参宏定义中,宏名和形参表之间不能有空格出现。 例如把: #define MAX(a,b) (ab)?a:b写为: #define MAX (a,b) (ab)?a:b将被认为是无参宏定义,宏名MAX代表字符串 (a,b) (ab)?a:b。宏展开时,宏调用语句: max=MAX(x,y);将变为: max=(a,b)(ab)?a:b(x,y);这显然是错误的。2)在带参宏定义中,形式参数不分配内存单元,因此不必作类型定义。在带参宏中,只是符号代换,不存在值传递的问题。3)在宏定义中的形参是标识符,而宏调用中的实参可以是表达式。【例】#include#define SQ(y) (
7、y)*(y)void main() int a,sq; printf(input a number: ); scanf(%d,&a); sq=SQ(a+1); printf(sq=%dn,sq);宏调用中实参为a+1,是一个表达式,在宏展开时,用a+1代换形参y,再用(y)*(y) 代换SQ,得到如下语句: sq=(a+1)*(a+1);4)定义宏时,宏串中把形参用圆括号括起来和不括是不同的,若宏定义改为:#define SQ(y) y*y则宏替换后为:sq=a+1*a+1;运行结果将完全不同。【例】#include#define SQ(y) (y)*(y)void main() int a,
8、sq; printf(input a number:); scanf(%d,&a); sq=160/SQ(a+1); printf(sq=%dn,sq);本程序与前例相比,只把宏调用语句改为: sq=160/SQ(a+1);运行本程序如输入值仍为3时,希望结果为10。但实际运行的结果如下:input a number:3sq=160分析宏调用语句,在宏代换之后变为: sq=160/(a+1)*(a+1);a为3时,由于“/”和“*”运算符优先级和结合性相同,则先作160/(3+1)得40,再作40*(3+1)最后得160。为了得到正确答案应在宏定义中的整个字符串外加括号,程序修改如下:【例】#
9、include#define SQ(y) (y)*(y)void main() int a,sq; printf(input a number:); scanf(%d,&a); sq=160/SQ(a+1); printf(sq=%dn,sq);对于宏定义不仅应在参数两侧加括号,也应在整个字符串外加括号。5)带参的宏和带参函数很相似,但有本质上的不同。【例1】#includeint SQ(int y) return(y)*(y);void main() int i=1; while(i=5) printf(%d:%dn,i,SQ(i+); 程序运行后输出:2:13:44:95:166:25【例
10、2】#include#define SQ(y) (y)*(y)void main() int i=1; while(i=5) printf(%d:%dn,i,SQ(i+); 程序运行输出:1:13:95:25在例1中,形参为Y,函数体表达式为(y)*(y)。在例2中。 形参也为y,字符串表达式为(y)*(y)。例1的函数调用为SQ(i+)例2的宏调用为SQ(i+),实参也是相同的。从输出结果来看,却大不相同。分析如下:在例1中,函数调用是把实参i值传给形参y后自增1。然后输出函数值。因而要循环5次。输出15的平方值。在例2中宏调用时,只作代换。SQ(i+)被代换为(i+)*(i+)。每一次循环
11、时,i完成乘运行后两次自增1,循环三次后,i=7,停止循环。从以上分析可以看出函数调用和宏调用二者在形式上相似,在本质上是完全不同的。6)宏定义也可用来定义多个语句,在宏调用时,把这些语句又代换到源程序内。看下面的例子。【例】#include#define SSSV(s1,s2,s3,v) s1=l*w;s2=l*h;s3=w*h;v=w*l*h;void main() int l=3,w=4,h=5,sa,sb,sc,vv; SSSV(sa,sb,sc,vv); printf(sa=%dnsb=%dnsc=%dnvv=%dn,sa,sb,sc,vv);6.3 文件包含文件包含是C预处理程序的
12、另一个重要功能。文件包含命令行的一般形式为: #include 文件名 在前面我们已多次用此命令包含过库函数的头文件。例如: #includestdio.h #includemath.h功能:是把指定的文件插入该命令行位置取代该命令行,从而把指定的文件和当前的源程序文件连成一个源文件。应用:一个大的程序可以分为多个模块,由多个程序员分别编程。有些公用的符号常量或宏定义等可单独组成一个文件,在其它文件的开头用包含命令包含该文件即可使用。这样,可避免在每个文件开头都去书写那些公用量,从而节省时间,并减少出错。说明:1)包含命令中的文件名可以用双引号括起来,也可以用尖括号括起来。例如以下写法都是允许
13、的:#includestdio.h#include但是这两种形式是有区别的:使用尖括号表示在包含文件目录(include)中去查找(包含目录是由用户在设置环境时设置的),而不在源文件目录去查找;使用双引号则表示首先在当前的源文件目录中查找,若未找到才到包含目录中去查找。2)一个include命令只能指定一个被包含文件,若有多个文件要包含,则需用多个include命令。3)文件包含允许嵌套,即在一个被包含的文件中又可以包含另一个文件。6.4 条件编译本章考点 宏定义。 文件包含。典型试题详解1有如下程序:#define N 2#define M N+1#define NUM 2*M+1main(
14、) int i;for(i=1;i=NUM;i+) printf(%dn,i);该程序中的for循环执行的次数是_。正确答案:6(知识点:不带参数的宏定义)2程序中头文件type1.h的内容是:#define N 5#define M1 N*3程序如下:#include type1.h#define M2 N*2main() int i; i=M1+M2; printf(%dn,i);程序编译后运行的输出结果是_。正确答案:25(知识点:文件包含、不带参数的宏定义)3以下程序的输出结果是_。#define SQR(X) X*Xmain()int a=16,k=2,m=1;a/=SQR(k+m)/SQR(k+m);printf(%dn,a);正确答案:2(知识点:带参数的宏定义)
copyright@ 2008-2022 冰豆网网站版权所有
经营许可证编号:鄂ICP备2022015515号-1