第7章编译预处理命會 2004年3月 河北建筑工程学院
第7章 编译预处理命令 2004年3月 河北建筑工程学院
编译预处理是C++编译系统的一个重要组成部分,它负 责分析处理几种特殊的指令,这些指令被称为预处理命令。 在C++源程序文件中,加入编译预处理命令,可以改进程 序设计环境,提高编程效率。但它们不是C++语言的组成 部分,不能直接对它们进行编译,编译系统在对源程序进 行正式的编译之前,必须先对这些命令进行预处理,经过 预处理后的程序不再包括预处理命令,然后由编译系统对 预处理后的源程序进行通常的编译处理,得到可供执行的 目标代码 C++提供的预处理命令主要有以下三种: 宏定义 文件包含 条件编译 这些命令均以#开头,每行一条命令,因为它们不是C++ 的语句,所以命令后无分号
编译预处理是C++编译系统的一个重要组成部分,它负 责分析处理几种特殊的指令,这些指令被称为预处理命令。 在C++源程序文件中,加入编译预处理命令,可以改进程 序设计环境,提高编程效率。但它们不是C++语言的组成 部分,不能直接对它们进行编译,编译系统在对源程序进 行正式的编译之前,必须先对这些命令进行预处理,经过 预处理后的程序不再包括预处理命令,然后由编译系统对 预处理后的源程序进行通常的编译处理,得到可供执行的 目标代码。 C++提供的预处理命令主要有以下三种: 宏定义 文件包含 条件编译 这些命令均以#开头,每行一条命令,因为它们不是C++ 的语句,所以命令后无分号
7.1宏定义 宏定义分为两类:不带参数的宏定义和带参数的宏定义。 711不带参数的宏定义 # define称为宏定义指令,它的一般格式为: # define标识符字符串 作用:用一个指定的标识符来代表一个字符串,字符串可以 是常量、关键字、语句、表达式,还可以是空字符 其中,标识符又称为宏名,字符串称为宏体。 编译预处理时,如果程序中出现宏定义命令,预处理程序就 把该命令以后的程序中所有同宏名一致的标识符全部替换为 所定义的宏体,这个过程称为“宏展开
7.1 宏定义 宏定义分为两类:不带参数的宏定义和带参数的宏定义。 7.1.1 不带参数的宏定义 #define称为宏定义指令,它的一般格式为: #define 标识符 字符串 作用:用一个指定的标识符来代表一个字符串,字符串可以 是常量、关键字、语句、表达式,还可以是空字符。 其中,标识符又称为宏名,字符串称为宏体。 编译预处理时,如果程序中出现宏定义命令,预处理程序就 把该命令以后的程序中所有同宏名一致的标识符全部替换为 所定义的宏体,这个过程称为“宏展开
例如: PI来代替字符串 3.1415926” #define pi 3.1415926 ※注意:在预编译处理时,将程序中在该命令以后 出现的所有的P都用“3.1415926”代替。 【例71】输入圆半径,求圆周长、圆面积、球体积。 cout cin>>re #define pi 3.1415926 =2*PIr; void maino s=PEror; =4.03.02 PIrrie; double ls, r,vi
例如: #define PI 3.1415926 PI来代替字符串 “3.1415926” ※ 注意: 在预编译处理时,将程序中在该命令以后 出现的所有的PI都用“3.1415926”代替。 【例7.1】 输入圆半径,求圆周长、圆面积、球体积。 #include #define PI 3.1415926 void main() { double l,s,r,v; cout>r; l=2*PI*r; s=PI*r*r; v=4.0/3.0*PI*r*r*r;
程序运行结果: input radius: 6 l=37.6991 s=113.097 v=904.779 说明: (1)宏名一般用大写字母表示,以便与变量名相区别。 2)使用宏名代替一个字符串,可以减少程序中重复书写 某些字符串的工作量,当需要改变某一个常量时,可以只 改变# define命令行,做到一改全改,不容易出错
程序运行结果: input radius: 6 l=37.6991 s=113.097 v=904.779 说明: (1) 宏名一般用大写字母表示,以便与变量名相区别。 (2) 使用宏名代替一个字符串,可以减少程序中重复书写 某些字符串的工作量,当需要改变某一个常量时,可以只 改变#define命令行,做到一改全改,不容易出错
(3)宏定义是用宏名代替一个字符串,在宏展开时只是作 简单的字符串替换,并不对语法是否正确进行检查。 例如在输入下列宏命令时: #define pi3. 1415926 将小数点“”错写成了“,”,在编译预处理时也不会报 错,只在编译时才会发现错误并报告错误。 (4)宏定义不是C++语句,一定不要在行末加分号,如果 加了分号,会将分号当成字符串的一部分进行替换。 (5)通常把# defin命令放在一个文件的开头,使其定义在 本文件内全部有效,即作用范围从其定义位置起到文件结 束 (6)可以使用# undef命令来取消宏定义的作用域
(3) 宏定义是用宏名代替一个字符串,在宏展开时只是作 简单的字符串替换,并不对语法是否正确进行检查。 例如在输入下列宏命令时: #define PI 3.1415926 将小数点“.”错写成了“ , ” ,在编译预处理时也不会报 错,只在编译时才会发现错误并报告错误。 (4) 宏定义不是C++语句,一定不要在行末加分号,如果 加了分号,会将分号当成字符串的一部分进行替换。 (5) 通常把#define命令放在一个文件的开头,使其定义在 本文件内全部有效,即作用范围从其定义位置起到文件结 束。 (6) 可以使用#undef命令来取消宏定义的作用域
# undef命令的一般格式为: # undef宏名 它的作用是通知编译预处理系统取消前面由# define命令所 定义的宏名,使其定义的宏名不再起作用。 例如: #define pi3.1415926 void maino #undef pi flo
#undef 命令的一般格式为: #undef 宏名 它的作用是通知编译预处理系统取消前面由#define命令所 定义的宏名,使其定义的宏名不再起作用。 例如: #define PI 3.1415926 void main() { …… } #undef PI f1() { ……}
由于# undef的作用,使PI的作用范围在# undef行处终止,因 此,在们1函数中,PI不再代表31415926。这样可以灵活控制 宏定义的作用范围。 (7)宏定义允许嵌套定义,即在进行宏定义时,可以使用已定 义过的宏名。 (8)对程序中用双引号括起来的字符串内的字符,即使与宏 名相同,也不进行替换 【例72】求半径为3的圆周长、圆面积和球体积。 #include define v4.0/3.0*PIR实 #definer 3 void main( #define pi 3.1415926 #define l 2 PlR cout<<"L="<<L<<"S="<<S <<"V="<<V<<endl;
由于#undef的作用,使PI的作用范围在#undef行处终止,因 此,在f1函数中,PI不再代表3.1415926。这样可以灵活控制 宏定义的作用范围。 (7) 宏定义允许嵌套定义,即在进行宏定义时,可以使用已定 义过的宏名。 (8) 对程序中用双引号括起来的字符串内的字符,即使与宏 名相同,也不进行替换 【例7.2】 求半径为3的圆周长、圆面积和球体积。 #include #define R 3 #define PI 3.1415926 #define L 2*PI*R #define V 4.0/3.0*PI*R*R*T void main() { cout<<"L="<<L<<"S="<<S <<" V="<<V<<endl; }
程序运行结果: L=18.8496 S=28.2743 V=113.097 (9)宏定义命令是专门用于预编译处理的,它与定义 变量的含义不同,只作字符替换,不分配内存空间。 7.12带参数的宏 这种宏不只是进行宏体的替换,还要进行参 数的替换
程序运行结果: L=18.8496 S=28.2743 V=113.097 (9) 宏定义命令是专门用于预编译处理的,它与定义 变量的含义不同,只作字符替换,不分配内存空间。 7.1.2 带参数的宏 这种宏不只是进行宏体的替换,还要进行参 数的替换
字符串中包含了在形式 一般形式: 参数表中所指定的参数 # define宏名(形式参数表)字符串 例如: #define s(a, b b area=S(3,2)得3*2 例73】计算圆面积 void maino #include #define pi 3. 1415926 float a, area; #define s(r)Pirr a=36;
一般形式 : #define 宏名(形式参数表)字符串 字符串中包含了在形式 参数表中所指定的参数 例如: #define S(a,b) a*b area = S(3,2) 得 3*2 【例7.3】 计算圆面积 #include #define PI 3.1415926 #define S(r) PI*r*r void main() { float a,area; a=3.6;