目录 8.1 宏定义 8.2 文件包含 8.3 条件编译 1
8.1 宏定义 1. 无参宏定义 1)无参宏定义的一般形式 #define 标识符 字符串 8.1 宏定义 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,然后再进行编译。 2
8.1 宏定义 1. 无参宏定义 1)无参宏定义的一般形式 为了避免宏代换时发生错误,宏定义中的字符串应加括号。 8.1 宏定义 1. 无参宏定义 1)无参宏定义的一般形式 为了避免宏代换时发生错误,宏定义中的字符串应加括号。 【例8_1】无参宏定义、宏调用与宏展开。 #include<stdio.h> #define m (y*y+3*y) void main() { int s,y; printf("input a number: ");scanf("%d",&y); s=3*m+4*m+5*m; printf("s=%d\n",s); } 3
8.1 宏定义 1. 无参宏定义 2)宏定义注意事项 (1)宏定义是用宏名来表示一个字符串,在宏展开时又以该字符串取代宏名,这只是一种简单的代换,字符串中可以含任何字符,可以是常数,也可以是表达式,预处理程序对它不作任何检查。如果宏定义有错误,只能在编译已被宏展开后的源文件时发现。 (2)宏定义不是声明或语句,在行末不必加分号,如加上分号则连分号也一起置换。 (3)宏定义必须写在函数之外,其作用域为从宏定义命令开始到源文件结束为止。如果要终止其作用域可使用# undef 命令。 4
8.1 宏定义 1. 无参宏定义 2)宏定义注意事项 (4)宏定义允许嵌套,在宏定义的字符串中可以使用已经定义的宏名。在宏展开时由预处理程序层层代换。例如: #define pi 3.1415926 #define s pi*y*y /* pi是已定义的宏名*/ 对语句:printf("%f",s); 在宏代换后变为: printf("%f",3.1415926*y*y); (5)习惯上,宏名用大写字母表示,以便于与变量区别;但也允许用小写字母。 5
8.1 宏定义 2)带参宏调用的一般形式 2. 带参宏定义 1)带参宏定义的一般形式 #define 宏名(形参表) 字符串 8.1 宏定义 2. 带参宏定义 1)带参宏定义的一般形式 #define 宏名(形参表) 字符串 2)带参宏调用的一般形式 宏名(实参表); 【例8_2】带参宏定义、宏调用与宏展开。 #include<stdio.h> #define max(a,b) (a>b)?a:b void main() { int x,y,max; printf("input two numbers: "); scanf("%d%d",&x,&y); max=max(x,y); printf("max=%d\n",max); } 6
8.1 宏定义 2. 带参宏定义 3)带参的宏定义注意事项 (1)带参宏定义中,宏名和形参表之间不能有空格出现。 8.1 宏定义 2. 带参宏定义 3)带参的宏定义注意事项 (1)带参宏定义中,宏名和形参表之间不能有空格出现。 (2)在带参宏定义中,形式参数不分配内存单元,因此不必作类型定义。而宏调用中的实参有具体的值,用实参替代形参,因此实参必须作类型声明;在函数中,形参和实参是两个不同的量,各有自己的作用域,调用时要把实参值赋给形参,进行值传递;而在带参宏中,只是符号代换,不存在值传递的问题。 (3)在宏定义中的形参只能是标识符,而宏调用中的实参可以是表达式。 7
8.1 宏定义 2. 带参宏定义 3)带参的宏定义注意事项 【例8_3】实参为表达式的宏调用。 8.1 宏定义 2. 带参宏定义 3)带参的宏定义注意事项 【例8_3】实参为表达式的宏调用。 #include<stdio.h> #define sq(y) (y)*(y) void main() { int a,sq; printf("input a number: "); scanf("%d",&a); sq=sq(a+1); printf("sq=%d\n",sq); } 8
8.1 宏定义 2. 带参宏定义 3)带参的宏定义注意事项 (4)为了避免宏代换时发生错误,宏定义中的字符串应加括号,字符中出现的形参也应加括号。 【例8_4】形参未加括号的错误。 #include<stdio.h> #define sq(y) y*y void main() { int a,sq; printf("input a number: "); scanf("%d",&a); sq=sq(a+1); printf("sq=%d\n",sq); } 9
8.1 宏定义 2. 带参宏定义 【例8_5】只形参加括号的错误。 3)带参的宏定义注意事项 8.1 宏定义 2. 带参宏定义 3)带参的宏定义注意事项 (4)为了避免宏代换时发生错误,宏定义中的字符串应加括号,字符中出现的形参也应加括号。 【例8_5】只形参加括号的错误。 #include<stdio.h> #define sq(y) (y)*(y) void main() { int a,sq; printf("input a number: "); scanf("%d",&a); sq=150/sq(a+1); printf("sq=%d\n",sq); } 10
8.1 宏定义 2. 带参宏定义 【例8_6】形参和字符串都加括号,无错误。 3)带参的宏定义注意事项 8.1 宏定义 2. 带参宏定义 3)带参的宏定义注意事项 (4)为了避免宏代换时发生错误,宏定义中的字符串应加括号,字符中出现的形参也应加括号。 【例8_6】形参和字符串都加括号,无错误。 #include<stdio.h> #define sq(y) ((y)*(y)) void main() { int a,sq; printf("input a number: "); scanf("%d",&a); sq=150/sq(a+1); printf("sq=%d\n",sq); } 11
8.1 宏定义 2. 带参宏定义 3)带参的宏定义注意事项 (5)带参的宏和带参函数很相似,但有本质上的不同,把同一表达式用函数处理与用宏处理两者的结果有可能是不同的。 【例8_7】参数传递。 #include<stdio.h> void main() { int i=1; while(i<=5) printf("%d\n",sq(i++)); } sq(int y) { return((y)*(y)); 12
8.1 宏定义 2. 带参宏定义 3)带参的宏定义注意事项 (5)带参的宏和带参函数很相似,但有本质上的不同,把同一表达式用函数处理与用宏处理两者的结果有可能是不同的。 【例8_8】宏替换。 #include<stdio.h> #define sq(y) ((y)*(y)) void main() { int i=1; while(i<=5) printf("%d\n",sq(i++)); } 13
8.2 文件包含 1. 文件包含的一般形式 2.文件包含的功能 #include"文件名"或#include<文件名> 8.2 文件包含 1. 文件包含的一般形式 #include"文件名"或#include<文件名> 两种形式的区别:使用尖括号表示在包含文件目录中去查找(包含目录是由用户在设置环境时设置的),而不在源文件目录去查找;使用双引号则表示首先在当前的源文件目录中查找,若未找到才到包含目录中去查找。用户编程时可根据自己文件所在的目录来选择某一种命令形式。 2.文件包含的功能 把指定的文件插入该命令位置取代该命令,从而把指定的文件和当前的源文件连成一个源文件。 在程序设计中,文件包含是很有用的。一个大的程序可以分为多个模块,由多个程序员分别编程。有些公用的符号常量或宏定义等可单独组成一个文件,在其它文件的开头用包含命令包含该文件即可使用。这样,可避免在每个文件开头都去书写那些公用量,从而节省时间,并减少出错。 14
8.2 文件包含 1. 文件包含使用注意事项 (1)一个include命令只能指定一个被包含文件,若有多个文件要包含,则需用多个include命令。 (2)文件包含允许嵌套,即在一个被包含的文件中又可以包含另一个文件。 15
8.3 条件编译 1.第一种形式 #ifdef 标识符 程序段1 #else 程序段2 #endif 8.3 条件编译 1.第一种形式 #ifdef 标识符 程序段1 #else 程序段2 #endif 它的功能是,如果标识符已被#define命令定义过则对程序段1进行编译;否则对程序段2进行编译。如果没有程序段2,本格式中的#else可以省略,即可以写为: 程序段 16
8.3 条件编译 1.第一种形式 【例8_9】条件编译的第一种形式。 #include<stdio.h> 8.3 条件编译 1.第一种形式 【例8_9】条件编译的第一种形式。 #include<stdio.h> #define num ok void main() { #ifdef num printf("Yes\n"); #else printf("No\n"); #endif } 17
8.3 条件编译 1.第二种形式 3.第三种形式 #ifndef 标识符 程序段1 #else 程序段2 #endif 8.3 条件编译 1.第二种形式 #ifndef 标识符 程序段1 #else 程序段2 #endif 功能:如果标识符未被#define命令定义过则对程序段1进行编译,否则对程序段2进行编译。 3.第三种形式 #if 常量表达式 功能:如常量表达式的值为真,则对程序段1进行编译,否则对程序段2进行编译。 18