第09章_预处理_第1页
第09章_预处理_第2页
第09章_预处理_第3页
第09章_预处理_第4页
第09章_预处理_第5页
已阅读5页,还剩26页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1、第九章第九章l 本章要点l 主要内容9.1 9.1 宏定义宏定义9.2“9.2“文件包含文件包含”处理处理9.3 9.3 条件编译条件编译4 基本概念基本概念 ANSI CANSI C标准规定可以在源程序中加入一些标准规定可以在源程序中加入一些“预处理命令预处理命令” ,以改进程序设计环境,提高,以改进程序设计环境,提高编程效率。编程效率。 这些预处理命令是由这些预处理命令是由ANSI CANSI C统一规定的,但是统一规定的,但是它不是它不是C C语言本身的组成部分,不能直接对它语言本身的组成部分,不能直接对它们进行编译(因为编译程序不能识别它们)。们进行编译(因为编译程序不能识别它们)。必

2、须在对程序进行通常的编译之前,先对程序必须在对程序进行通常的编译之前,先对程序中这些特殊的命令进行中这些特殊的命令进行“预处理预处理” 经过预处理后程序可由编译程序对预处理后的经过预处理后程序可由编译程序对预处理后的源程序进行通常的编译处理,得到可供执行的源程序进行通常的编译处理,得到可供执行的目标代码。目标代码。 5 基本概念基本概念 C C语言与其他高级语言的一个重要区别是可以语言与其他高级语言的一个重要区别是可以使用预处理命令和具有预处理的功能。使用预处理命令和具有预处理的功能。提供的预处理功能主要有以下三种:提供的预处理功能主要有以下三种:宏定义宏定义文件包含文件包含条件编译条件编译

3、这些功能分别用宏定义命令、文件包含命这些功能分别用宏定义命令、文件包含命令、条件编译命令来实现。为了与一般语句令、条件编译命令来实现。为了与一般语句相区别,这些命令以符号相区别,这些命令以符号“”开头。例如:开头。例如: #define #define #include #include 6 9.1 9.1 宏定义宏定义9.1.1 9.1.1 不带参数的宏定义不带参数的宏定义 宏定义一宏定义一般形式为般形式为: :define 标识符 字符串例如:define PI 3.1415926 宏定义的作用是在本程序文件中用指定的标识符宏定义的作用是在本程序文件中用指定的标识符PIPI来代替来代替“3

4、.14159263.1415926”这个字符串,在编译预处理这个字符串,在编译预处理时,将程序中在该命令以后出现的所有的时,将程序中在该命令以后出现的所有的PIPI都用都用“3.14159263.1415926”代替。这种方法使用户能以一个简代替。这种方法使用户能以一个简单的名字代替一个长的字符串单的名字代替一个长的字符串. . 这个标识符(名字)称为这个标识符(名字)称为“宏名宏名” 在预编译时将宏名替换成字符串的过程称为在预编译时将宏名替换成字符串的过程称为“宏展宏展开开”。definedefine是宏定义命令是宏定义命令。7#include stdio.h#include #define

5、 PI 3.1415926#define PI 3.1415926void main()void main() float l,s,r,vfloat l,s,r,v; ; printf(input printf(input radius:); radius:); scanf(%f,&r scanf(%f,&r);); l=2.0 l=2.0* *PIPI* *r;r; s= s=PIPI* *r r* *r;r; v=4.0/3 v=4.0/3* *PIPI* *r r* *r r* *r;r; printf(l printf(l=%10.4fns=%10.4fnv=%10.4

6、fn,l,s,v);=%10.4fns=%10.4fnv=%10.4fn,l,s,v); 例例9.1 9.1 使用不带参数的宏定义使用不带参数的宏定义8input radius: input radius: 4 运行情况如下:运行情况如下:1=25.13281=25.1328s=50.2655s=50.2655v=150.7966v=150.7966(1) (1) 宏名一般习惯用大写字母表示,以便与变量名宏名一般习惯用大写字母表示,以便与变量名相区别。但这并非规定,也可用小写字母。相区别。但这并非规定,也可用小写字母。(2) (2) 使用宏名代替一个字符串,可以减少程序中重使用宏名代替一个字符

7、串,可以减少程序中重复书写某些字符串的工作量。复书写某些字符串的工作量。(3) (3) 宏定义是用宏名代替一个字符串,只作简单置宏定义是用宏名代替一个字符串,只作简单置换,不作正确性检查。只有在编译已被宏展开后换,不作正确性检查。只有在编译已被宏展开后的源程序时才会发现语法错误并报错。的源程序时才会发现语法错误并报错。9(4) (4) 宏定义不是语句,不必在行末加分号。如果宏定义不是语句,不必在行末加分号。如果加了分号则会连分号一起进行置换。加了分号则会连分号一起进行置换。(5) (5) definedefine命令出现在程序中函数的外面,宏名命令出现在程序中函数的外面,宏名的有效范围为定义命

8、令之后到本源文件结束。通的有效范围为定义命令之后到本源文件结束。通常,常,definedefine命令写在文件开头,函数之前,作命令写在文件开头,函数之前,作为文件一部分,在此文件范围内有效。为文件一部分,在此文件范围内有效。(6) (6) 可以用可以用unundefdef命令终止宏定义的作用域。命令终止宏定义的作用域。例如:例如: 10#define G 9.8 #define G 9.8 _ void main() void main() G G的有效范围的有效范围 - #undef #undef G G f1() f1() 在在f1f1函数中,不再代表函数中,不再代表9.89.8。这样可

9、以灵活控制宏。这样可以灵活控制宏定义的作用范围。定义的作用范围。11(7) (7) 在进行宏定义时,可以引用已定义的宏名,可在进行宏定义时,可以引用已定义的宏名,可以层层置换。以层层置换。 #include stdio.h#include #define R 3.0#define R 3.0#define PI 3.1415926#define PI 3.1415926#define L 2#define L 2* *PIPI* *R R#define S #define S PIPI* *R R* *R Rvoid main()void main() printf( printf(L L=%

10、fn=%fnS S=%fn,=%fn,L L, ,S S);); 运行情况如下:运行情况如下:L=18.849556L=18.849556S=28.274333S=28.274333例例9.2 9.2 在宏定义中引用已定义的宏名在宏定义中引用已定义的宏名12经过宏展开后,经过宏展开后,printfprintf函数中的输出项被展开为函数中的输出项被展开为: : 2 2* *3.14159263.1415926* *3.03.0展开为展开为 3.14159263.1415926* *3.03.0* *3.03.0printfprintf函数调用语句展开为函数调用语句展开为: :printf(L=%

11、FNS=%fn,2*3.1415926*3.0,3.1415926*3.0*3.0);13(8) (8) 对程序中用双撇号括起来的字符串内的字符,即对程序中用双撇号括起来的字符串内的字符,即使与宏名相同,也不进行置换。使与宏名相同,也不进行置换。 (9) (9) 宏定义是专门用于预处理命令的一个专用名词,宏定义是专门用于预处理命令的一个专用名词,它与定义变量的含义不同,只作字符替换,不分配它与定义变量的含义不同,只作字符替换,不分配内存空间。内存空间。14 9.1.2 9.1.2 带参数的宏定义带参数的宏定义 作用:作用:不是进行简单的字符串替换,还要不是进行简单的字符串替换,还要进行参数替换

12、。进行参数替换。 带参数的宏定义一般形式为带参数的宏定义一般形式为: :define define 宏名(参数表)宏名(参数表) 字符串字符串 字符串中包含在括弧中所指定的参数字符串中包含在括弧中所指定的参数#define #define S(a,b S(a,b) a) a* *b b area=S(3,2);area=S(3,2); 程序中用和分别代替宏程序中用和分别代替宏定义中的形式参数和定义中的形式参数和b b,用用* *代替代替S(3,2)S(3,2) 。因此。因此赋值语句展开为:赋值语句展开为:area=3area=3* *2 2例:15 对带实参的宏对带实参的宏(如(如S S(3

13、3,2 2),),则按则按definedefine命令命令行中指定的字符串从左到行中指定的字符串从左到右进行置换。若串中包含右进行置换。若串中包含宏中的形参(如宏中的形参(如a a、b b),),则将程序中相应的实参(则将程序中相应的实参(可以是常量、变量或表达可以是常量、变量或表达式)代替形参。如果宏定式)代替形参。如果宏定义中的字符串中的字符不义中的字符串中的字符不是参数字符(如是参数字符(如* *中中的的* *号),则保留。这样号),则保留。这样就形成了置换的字符串。就形成了置换的字符串。对带参的宏定义是这样展开置换的:对带参的宏定义是这样展开置换的: 16#include stdio.

14、h#include #define PI 3.1415926#define PI 3.1415926#define S(r#define S(r) PI) PI* *r r* *r rvoid main()void main() float a,areafloat a,area; ; a=3.6;a=3.6; area=area=S(aS(a) ); ; printf(rprintf(r=%fnarea=%fnarea=%fn,a,area=%fn,a,area);); 运行情况如下:运行情况如下:=3.600000=3.600000 area=40.715038 area=40.715038

15、例例9.3 9.3 使用带参的宏使用带参的宏赋值语句赋值语句“area=S(aarea=S(a);); ” ” 经宏展开后为:经宏展开后为:area=3.1415926area=3.1415926* * *; 1718(1)(1)对带参数的宏展开只是将语句中的宏名后面对带参数的宏展开只是将语句中的宏名后面括号内的实参字符串代替括号内的实参字符串代替definedefine 命令行命令行中的形参。中的形参。(2) (2) 在宏定义时,在宏名与带参数的括弧之间不在宏定义时,在宏名与带参数的括弧之间不应加空格,否则将空格以后的字符都作为应加空格,否则将空格以后的字符都作为替代字符串的一部分。替代字符

16、串的一部分。 19(1) (1) 函数调用时,先求出实参表达式的值,然后代入形函数调用时,先求出实参表达式的值,然后代入形参。而使用带参的宏只是进行简单的字符替换。参。而使用带参的宏只是进行简单的字符替换。(2) (2) 函数调用是在程序运行时处理的,为形参分配临时函数调用是在程序运行时处理的,为形参分配临时的内存单元。而宏展开则是在编译前进行的,在展的内存单元。而宏展开则是在编译前进行的,在展开时并不分配内存单元,不进行值的传递处理,也开时并不分配内存单元,不进行值的传递处理,也没有没有“返回值返回值”的概念。的概念。(3) (3) 对函数中的实参和形参类型要求一致。而宏名无类对函数中的实参

17、和形参类型要求一致。而宏名无类型,它的参数也无类型,只是一个符号代表,展开型,它的参数也无类型,只是一个符号代表,展开时代入指定的字符串即可。宏定义时,字符串可以时代入指定的字符串即可。宏定义时,字符串可以是任何类型的数据。是任何类型的数据。(4) (4) 调用函数只可得到一个返回值,而用宏可以设法得调用函数只可得到一个返回值,而用宏可以设法得到几个结果。到几个结果。 20#include stdio.h#include #define PI 3.1415926#define PI 3.1415926#define CIRCLE(R,L,S,V)#define CIRCLE(R,L,S,V)

18、L=2 L=2* *PIPI* *R; S=PIR; S=PI* *R R* *R; R; V=4.0/3.0V=4.0/3.0* *PIPI* *R R* *R R* *R Rvoid main()void main() float r,l,s,vfloat r,l,s,v; ; scanf(%f,&rscanf(%f,&r);); CIRCLE(r,l,s,vCIRCLE(r,l,s,v);); printf(rprintf(r=%6.2f,l=%6.2f,s=%6.2f,v=%6.2fn,=%6.2f,l=%6.2f,s=%6.2f,v=%6.2fn,r,l,s,v);r

19、,l,s,v); 例例9.4 9.4 通过宏展开得到若干个结果通过宏展开得到若干个结果21void mainvoid main()() float r,l,s,vfloat r,l,s,v; ; scanf(%f,&rscanf(%f,&r);); l=2l=2* *3.14159263.1415926* *r; r; s=3.1515926s=3.1515926* *r r* *r;r; v=4.0/3/0v=4.0/3/0* *3.14159263.1415926* *r r* *r r* *r;r; printf(”rprintf(”r=%6.2f,l=%6.2f,s=%

20、6.2f,v=%6.2f=%6.2f,l=%6.2f,s=%6.2f,v=%6.2fn”,r,l,s,v);n”,r,l,s,v); 对宏进行预编译,展开后的对宏进行预编译,展开后的mainmain函数如下:函数如下:运行情况如下:运行情况如下:3.53.5r r=3.50=3.50,l=21.99l=21.99,s=38.48s=38.48,v=179.59v=179.5922(5) (5) 使用宏次数多时,宏展开后源程序长,因为每使用宏次数多时,宏展开后源程序长,因为每展开一次都使程序增长,而函数调用不会使源展开一次都使程序增长,而函数调用不会使源程序变长。程序变长。(6) (6) 宏替换

21、不占运行时间,只占编译时间。而函数宏替换不占运行时间,只占编译时间。而函数调用则占运行时间(分配单元、保留现场、值调用则占运行时间(分配单元、保留现场、值传递、返回)。传递、返回)。如果善于利用宏定义,可以实现程序的简化如果善于利用宏定义,可以实现程序的简化,如事先将程序中的,如事先将程序中的“输出格式输出格式”定义好,以定义好,以减少在输出语句中每次都要写出具体的输出格减少在输出语句中每次都要写出具体的输出格式的麻烦。式的麻烦。23例例9.5 9.5 通过宏展开得到若干个结果通过宏展开得到若干个结果#include #define PR printf#define NL n#define D

22、 %d#define D1 D NL#define D2 D D NL#define D3 D D D NL#define D4 D D D D NL#define S %svoid main() int a,b,c,d; char string=CHINA; a=1;b=2;c=3;d=4; PR(D1,a); PR(D2,a,b); PR(D3,a,b,c); PR(D4,a,b,c,d); PR(S,string); 运行时输出结果:运行时输出结果:24 所谓所谓“文件包含文件包含”处理是指一个源文件可以将另处理是指一个源文件可以将另外一个源文件的全部内容包含进来。语言提供外一个源文件的

23、全部内容包含进来。语言提供了了#include#include命令用来实现命令用来实现“文件包含文件包含”的操作。的操作。其一般形式为其一般形式为: : #include #include 文件名文件名 或或 #include #include 9.2 “9.2 “文件包含文件包含”处理处理25例例9.6 9.6 将例将例9 95 5时格式宏做成头文件,把它时格式宏做成头文件,把它包含在用户程序中。包含在用户程序中。(1)将格式宏做成头文件format.h #include #define PR printf#define NL n#define D %d#define D1 D NL#def

24、ine D2 D D NL#define D3 D D D NL#define D4 D D D D NL#define S %s(2)主文件file1.c#include #include format.hvoid main() int a,b,c,d; char string=CHINA; a=1;b=2;c=3;d=4; PR(D1,a); PR(D2,a,b); PR(D3,a,b,c); PR(D4,a,b,c,d); PR(S,string); 26注意:注意: 在编译时并不是分别对两个文件分别进行编在编译时并不是分别对两个文件分别进行编译,然后再将它们的目标程序连接的,而是译,然

25、后再将它们的目标程序连接的,而是在经过编译预处理后将头文件在经过编译预处理后将头文件format.hformat.h包含包含到主文件中,得到一个新的源程序,然后对到主文件中,得到一个新的源程序,然后对这个文件进行编译,得到一个目标(这个文件进行编译,得到一个目标(.obj.obj)文件。被包含的文件成为新的源文件的一部文件。被包含的文件成为新的源文件的一部分,而单独生成目标文件。分,而单独生成目标文件。27(1) (1) 一个一个#include#include命令只能指定一个被包含文件,命令只能指定一个被包含文件,如果要包含个文件,要用个如果要包含个文件,要用个#include#includ

26、e命令。命令。(2) (2) 如果文件包含文件,而在文件中要用到如果文件包含文件,而在文件中要用到文件的内容,则可在文件中用两个文件的内容,则可在文件中用两个includeinclude命令分别包含文件和文件,而且文件应出命令分别包含文件和文件,而且文件应出现在文件之前,即在现在文件之前,即在file1.cfile1.c中定义。中定义。(3) (3) 在一个被包含文件中又可以包含另一个被包含在一个被包含文件中又可以包含另一个被包含文件,即文件包含是可以嵌套的。文件,即文件包含是可以嵌套的。 2829(4) (4) 在在#include#include命令中,文件名可以用双撇号或尖命令中,文件名可以用双撇号或尖括号括起来。括号括起来。(5) (5) 被包含文件(被包含文件(file2.hfile2.h)与其所在的文件(即)与其所在的文件(即用用#include#include命令的源文件命令的源文件file2.cfile2.c),在预编译),在预编译后已成为

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论