C程序设计 谭浩强 著 清华大学出版社
C程序设计 第九章 预处理命令 概述 预处理是C语言的一个重要功能,它由预处理程序负责完成。当对一个源文件进行编译时,系统将自动引用预处理程序对源程序中的预处理部分作处理,处理完毕自动进入对源程序的编译。 以“#”号开头的预处理命令。如包含命令#include,宏定义命令#define等。在源程序中这些命令都放在函数之外,而且一般都放在源文件的前面,它们称为预处理部分。
C程序设计 第九章 预处理命令 目录 宏定义 文件包含 条件编译 本章小结 本章结束
9.2 宏定义 9.2.1 无参宏定义 无参宏的宏名后不带参数。 其定义的一般形式为: #define 标识符 字符串 【例9.1】 C程序设计 第九章 预处理命令 9.2 宏定义 9.2.1 无参宏定义 无参宏的宏名后不带参数。 其定义的一般形式为: #define 标识符 字符串 【例9.1】 【例9.2】 例如:#define PI 3.1415926 其中的“#”表示这是一条预处理命令。凡是以“#”开头的均为预处理命令。“define”为宏定义命令。“标识符”为所定义的宏名。“字符串”可以是常数、表达式、格式串等。
9.2 宏定义 9.2.2 带参宏定义 对带参数的宏,在调用中,不仅要宏展开,而且要用实参去代换形参。 带参宏定义的一般形式为: C程序设计 第九章 预处理命令 9.2 宏定义 9.2.2 带参宏定义 对带参数的宏,在调用中,不仅要宏展开,而且要用实参去代换形参。 带参宏定义的一般形式为: #define 宏名(形参表) 字符串 如:#define S(a,b) a*b area=S(3,2); 在字符串中含有各个形参。 带参宏调用的一般形式为: 宏名(实参表); 【例9.3】 返回主菜单
例9.1 #define PI 3.1415926 main() {float l,s,r,v; C程序设计 第九章 预处理命令 例9.1 #define PI 3.1415926 main() {float l,s,r,v; printf("input radius:"); scanf("%f",&r); l=2.0*PI*r; s=PI*r*r;v=4.0/3*PI*r*r*r; printf("l=%10.4f\ns=%10.4f\nv=%10.4f\n",l,s,v); } 输出: input radius:4 l= 25.1327 s= 50.2655 v= 268.0826 说明 运行程序
无参宏定义说明 1)宏名一般习惯用大写字母表示,以便与变量名相区别。但这并非规定,也可用小写字母。 2)宏定义是用宏名来表示一个字符串。 3)在宏展开时又以该字符串取代宏名,这只是一种简单的代换,字符串中可以含任何字符,可以是常数,也可以是表达式,预处理程序对它不作任何检查。如有错误,只能在编译已被宏展开后的源程序时发现。 4)宏定义不是说明或语句,在行末不必加分号,如加上分号则连分号也一起置换。
无参宏定义说明(续) 5)#define命令出现在程序中函数的外面,宏名的有效范围为定义命令之后到本源文件结束,通常#define命令写在文件开头。 6)宏定义必须写在函数之外,其作用域为宏定义命令起到源程序结束。如要终止其作用域可使用# undef命令。 例如: #define PI 3.14159 main() { …… } #undef PI f1() 表示PI只在main函数中有效,在f1中无效。
无参宏定义说明(续) 7)在进行宏定义时,可以引用已定义的宏名,可以层层置换。 8)对程序中引用双撇号括起来的字符串内的字符,即使与宏名相同,也不进行置换。 9)宏定义是专门用于预处理命令的一个专门名词,它与定义变量的含义不同,只作字符替换,不分配内存空间。
例9.2 #define R 3.0 #define PI 3.1415926 #define L 2*PI*R C程序设计 第九章 预处理命令 例9.2 #define R 3.0 #define PI 3.1415926 #define L 2*PI*R #define S PI*R*R main() { printf("L=%f\nS=%f\n",L,S); } 输出: L=18.849556 S=28.274333 运行程序
例9.3 #define PI 3.1415926 #define S(r) PI*r*r main() {float a,area; C程序设计 第九章 预处理命令 例9.3 #define PI 3.1415926 #define S(r) PI*r*r main() {float a,area; a=3.6; area=S(a); printf("r=%f\narea=%f\n",a,area); } 说明 输出: r=3.600000 area=40.715038 运行程序
带参宏定义说明 对于带参的宏定义有以下问题需要说明: 1)对带参数的宏展开只是将语句中的宏名后面括号内的实参字符串代替#difine命令行中的形参。 2)在宏定义时,宏名和形参表之间不能有空格出现。 3)对函数中的实参和形参都要定义类型,二者的类型要求一致,如不一致,应进行类型转换。而宏不存在类型问题,宏名无类型,它的参数也无类型,只是一个符号代表,展开时代入指定的字符串即可,宏定义时,字符串可以是任意类型的数据。
带参宏定义说明(续) 4)调用函数只可得到一个返回值,而用宏可以设法得到几个结果。 【例9.4】 5)使用宏次数多时,宏展开后源程序长,因为每展开一次都使程序增长,而函数调用不使源程序变长。 6)宏替换不占运行时间,只占编译时间,而函数调用则占运行时间(分配单元、保留现场,值传递、返回)。 【例9.5】
C程序设计 第九章 预处理命令 例9.4 #define PI 3.1415926 #define CIRCLE(R,L,S,V) L=2*PI*R;S=PI*R*R;V=4.0/3.0*PI*R*R*R main() {float r,l,s,v; scanf("%f",&r); CIRCLE(r,l,s,v); printf("r=%6.2f,l=%6.2f,s=%6.2f,v=%6.2f\n",r,l,s,v); } 输入:3.5 输出: r=3.50,l=21.99,s=38.48,v=179.59 运行程序
例9.5 输出: 1 12 123 1234 CHINA 运行程序 #define PR printf #define NL "\n" #define D"%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 "%s" 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); } 输出: 1 12 123 1234 CHINA 运行程序
9.3 文件包含 文件包含是C预处理程序的另一个重要功能。 文件包含命令行的一般形式为: #include"文件名" 9.3 文件包含 文件包含是C预处理程序的另一个重要功能。 文件包含命令行的一般形式为: #include"文件名" 在前面我们已多次用此命令包含过库函数的头文件。例如: #include "stdio.h" #include "math.h" 【例9.6】 返回主菜单
C程序设计 第九章 预处理命令 9.4 条件编译 预处理程序提供了条件编译的功能。可以按不同的条件去编译不同的程序部分,因而产生不同的目标代码文件。这对于程序的移植和调试是很有用的。条件编译有三种形式: 第一种形式: #ifdef 标识符 程序段1 #else 程序段2 #endif #ifdef 标识符 程序段 #endif 它的功能是,如果标识符已被 #define命令定义过则对程序段1进行编译;否则对程序段2进行编译。如果没有程序段2(它为空),本格式中的#else可以没有,即可以写为:
C程序设计 第九章 预处理命令 9.4 条件编译 第二种形式: #ifndef 标识符 程序段1 #else 程序段2 #endif 与第一种形式的区别是将“ifdef”改为“ifndef”。它的功能是,如果标识符未被#define命令定义过则对程序段1进行编译,否则对程序段2进行编译。这与第一种形式的功能正相反。
例9.6 文件 file1.c 文件 format.h #include "format.h" 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); } #define PR printf #define NL "\n" #define D"%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 "%s" 输出: 1 12 123 1234 CHINA 说明 运行程序
文件包含说明 1)一个include 命令只能指定一个被包含文件,如果要包含n个文件,要用n个#include命令。 2)如果文件1包含文件2,而文件2中要用到文件3内容,则可在文件1中用两个include命令分别包含文件2和文件3,而文件3应出现在文件2之前,即在file1.c中定义: #include “file3.h” #include “file2.h” 3)在一个被包含文件中又可以包含另一个被包含文件,即文件包含是可以嵌套的。
文件包含说明(续) 4)在#include命令中,文件名可以用双撇号匮尖括号括起来。如可以在file1.c中用: #include <file2.h> 或include “file2.h” 5)被包含文件(file2.h)与其所在的文件(即用#include命令的源文件file1.c)在预编译后已成为同一个文件,因此,如果file2.h中有全局静态变量,它也在file1.c文件中有效,不必用extern声明。
C程序设计 第九章 预处理命令 9.4 条件编译 它的功能是,如常量表达式的值为真(非0),则对程序段1 进行编译,否则对程序段2进行编译。因此可以使程序在不同条件下,完成不同的功能。 第三种形式: #if 常量表达式 程序段1 #else 程序段2 #endif 【例9.7】 返回主菜单
例9.7 输出: C LANGUAGE 运行程序 #define LETTER 1 main() {char str[20]="C Language",c; int i; i=0; while((c=str[i])!='\0') {i++; #if LETTER if(c>='a'&&c<='z') c=c-32; #else if(c>='A'&&c<='Z') c=c+32; #endif printf("%c",c); } 输出: C LANGUAGE 运行程序
9.5 本章小结 预处理功能是C语言特有的功能,它是在对源程序正式编译前由预处理程序完成的。程序员在程序中用预处理命令来调用这些功能。 C程序设计 第九章 预处理命令 9.5 本章小结 预处理功能是C语言特有的功能,它是在对源程序正式编译前由预处理程序完成的。程序员在程序中用预处理命令来调用这些功能。 宏定义是用一个标识符来表示一个字符串,这个字符串可以是常量、变量或表达式。在宏调用中将用该字符串代换宏名。 宏定义可以带有参数,宏调用时是以实参代换形参。而不是“值传送”。 为了避免宏代换时发生错误,宏定义中的字符串应加括号,字符串中出现的形式参数两边也应加括号。
9.5 本章小结 文件包含是预处理的一个重要功能,它可用来把多个源文件连接成一个源文件进行编译,结果将生成一个目标文件。 C程序设计 第九章 预处理命令 9.5 本章小结 文件包含是预处理的一个重要功能,它可用来把多个源文件连接成一个源文件进行编译,结果将生成一个目标文件。 条件编译允许只编译源程序中满足条件的程序段,使生成的目标程序较短,从而减少了内存的开销并提高了程序的效率。 使用预处理功能便于程序的修改、阅读、移植和调试,也便于实现模块化程序设计。 返回主菜单
The end