1、第7章 编译预处理学习目标 领会编译预处理的作用和基本过程。会使用常用的编译预处理命令。知道运用编译预处理命令时的常见错误。编译预处理的概念 7.1 宏定义 7.2 文件包含 7.3 条件编译什么是编译预处理?预处理命令用在什么位置?什么是宏定义?什么是宏替换?宏定义时如何使用参数?什么文件包含?常用的头文件有哪些?怎样使用条件编译命令?条件编译命令有哪些格式?所谓预处理是指在进行编译的第一遍扫描(词法扫描和语法分析)之前所作的工作。在源程序中,这些命令一般都放在源文件的前面,称为预处理部分。返回7.1 宏定义 在语言源程序中允许用一个标识符来表示一个字符串,称为“宏”。宏定义是由源程序中的宏
2、定义命令完成的。宏替换是由预处理程序自动完成的。7.1.1 无参宏定义 其定义的一般形式为:#define 标识符 字符串例如:#define W(x*x+x+1)如:#define W(x*x+x+1)main()int x,y;printf(input a number:);scanf(%d,&x);y=W*W+5*W;printf(y=%dn,y);宏定义说明:(1)宏定义是用宏名来表示一个字符串,在宏展开时又以该字符串取代宏名。字符串中可以含任何字符,可以是常数,也可以是表达式,甚至可以是一个完整的语句。但应注意的是,不能试图将编译预处理命令本身定义为宏。如:#define D def
3、ine#D PI 3.1415926 /*无法完成宏替换*/(2)预处理程序仅对程序中的宏作字符替换,而不作任何检查。如有错误,只能在编译宏展开后的源程序时发现。(3)宏定义不是说明或语句,因而在行末不必加分号,如加上分号则在宏展开时连同分号一起替换。宏定义说明:(4)宏定义必须写在函数之外,其作用域为宏定义命令起到源程序结束。如果需要终止其作用域可使用#undef命令。例如:宏定义说明:#define PI 3.14159main()#undef PIf1()(5)在函数体的字符串常量和定义过的变量名中若出现宏名,则编译预处理程序不对其作宏替换。如:宏定义说明:#define INTNUM
4、32767#define s 314main()int s1=1;printf(INTNUMn);printf(%dn,s);printf(%dn,s1);运行结果:INTNUM3141(6)宏定义允许嵌套,即:在宏定义的字符串中可以使用已经定义的宏名。在宏展开时由预处理程序层层替换。例如:宏定义说明:#define PI 3.1415926#define S PI*r*r /*PI是已定义的宏名*/对语句:printf(%f,S);在宏替换后变为:printf(%f,3.1415926*r*r);(7)习惯上宏名用大写字母表示,以便于与变量区别。但在语法上允许用小写字母。(8)可用宏定义表示
5、数据类型,以便采用习惯或简便的写法。例如:宏定义说明:#define STU struct stu在程序中可用STU作变量说明:STU student5,*p;(9)对“输出格式”作宏定义,可以减少书写麻烦。宏定义说明:#define P printf#define D%dn#define F%fnmain()int a=3,b=6,c=9;float e=3.8,f=6.9,g=2.98;P(D F,a,e);P(D F,b,f);P(D F,c,g);如:带参宏定义的一般形式为:#define 宏名(形参表)字符串在字符串中含有形参表中列出的各个形参。带参宏调用的一般形式为:宏名(实参表)
6、;7.1.2 带参宏定义例如:#define N(y)y*y+y+1/*宏定义*/T=N(5);/*宏调用*/例 7.1 带参宏替换。#define MAX(a,b)(ab)?a:bmain()int x,y,max;printf(input two numbers:);scanf(%d%d,&x,&y);max=MAX(x,y);printf(max=%dn,max);带参宏定义说明:(1)带参宏定义中,宏名和形参表之间不能有空格出现。例如:#define MAX(a,b)(ab)?a:b如果写成:#define MAX(a,b)(ab)?a:b (2)在带参宏定义中,形式参数不分配内存单元
7、,因而不必作类型定义。而宏调用中的实参有具体的值。要用它们去替换形参,因而必须作类型说明。这是与函数调用时情况有所不同。在函数调用过程中,形参和实参是两个不同的量,有不同的作用域,调用时要把实参值赋予形参,进行“值传递”。而在带参宏中,只是符号替换,不存在值传递的问题。带参宏定义说明:(3)在宏定义中的形参是标识符,而宏调用中的实参可以是表达式。例:带参宏定义说明:#define SQ(y)(y)*(y)main()int a,sq;printf(input a number:);scanf(%d,&a);sq=SQ(a+1);printf(sq=%dn,sq);(4)在宏定义中,字符串内的形
8、参通常要用括号括起来以避免出错。(5)带参的宏和带参函数很相似,但有本质上的不同,除上面已谈到的各点外,把同一表达式用函数处理与用宏处理两者的结果有可能是不同的。带参宏定义说明:(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=%dnsb=%dnsc=%dnvv=%dn,sa,sb,sc,vv);返回7.2 文件包含 文件
9、包含命令行的一般形式为:#include文件名“文件包含命令的功能是:把指定的文件插入该命令行位置取代该命令行,从而把指定的文件和当前的源程序文件连成一个源文件。文件包含命令说明:(1)包含命令中的文件名可以用双引号括起来,也可以用尖括号括起来。但是这两种形式是有区别的。#includestdio.h#include使用尖括号表示在包含文件目录中去查找(包含目录是由用户在设置环境时设置的),而不在源文件目录去查找;使用双引号则表示首先在当前的源文件目录中查找,若未找到才到包含目录中去查找。文件包含命令说明:(2)一个include命令只能指定一个被包含文件,若有多个文件要包含,则需用多个inc
10、lude命令。(3)文件包含允许嵌套,即在一个被包含的文件中又可以包含另一个文件。返回7.3 条件编译 1.第一种形式:功能:如果标识符已被#define命令定义过,则对程序段1进行编译;否则对程序段2进行编译。#ifdef 标识符程序段1#else程序段2#endif/*条件编译举例*/2.第二种形式:功能:如果标识符未被#define命令定义过则对程序段1进行编译,否则对程序段2进行编译。这与第一种形式的功能正相反。#ifndef 标识符程序段1#else程序段2#endif 3.第三种形式:功能:如常量表达式的值为真(非0),则对程序段1 进行编译,否则对程序段2进行编译。#if 常量表达式程序段1#else程序段2#endif举例: