第6章编译预处理 ★作用:对源程序编译之前做一些处理,生成扩展C 源程序 ★种类 今宏定义 define ◆文件包含# include ◆条件编译#f--#else-#endf等 ★格式 ◆“#开头 今占单独书写行 心语旬尾不加分号
第6章 编译预处理 作用:对源程序编译之前做一些处理,生成扩展C 源程序 种类 ❖宏定义 #define ❖文件包含 #include ❖条件编译 #if--#else--#endif等 格式: ❖“#”开头 ❖占单独书写行 ❖语句尾不加分号
§7.1宏定义 宏体可缺省,表示宏名 定义过或取消宏体 ★不带参数宏定义 ☆一般形式:# define宏名[宏体] ☆功能用指定标识符(宏名)代替字符序列(宏体) 冷望定/凝在函数外面) define o 0 ☆作用始义會令到夜猎续束 ☆山nd的止身倒和 efine YES1 格例# define WidTH80 YES原作用域 令宏展# efine LENGtH WidtH+40作语法检查 var=LENGTH2 号宏展开:var=80+402 不难0 YES新作用域 吸蜘炎耐 玄齐:prm2干%m,3午412
如 if(x==YES) printf(“correct!\n”); else if (x==NO) printf(“error!\n”); 展开后: if(x==1) printf(“correct!\n”); else if (x==0) printf(“error!\n”); §7.1 宏定义 不带参数宏定义 ❖一般形式: #define 宏名 [宏体] ❖功能:用指定标识符(宏名)代替字符序列(宏体) ❖宏展开:预编译时,用宏体替换宏名---不作语法检查 如 #define YES 1 #define NO 0 #define PI 3.1415926 #define OUT printf(“Hello,World”); 宏体可缺省,表示宏名 定义过或取消宏体 ❖定义位置:任意(一般在函数外面) ❖作用域:从定义命令到文件结束 ❖#undef可终止宏名作用域 格式: #undef 宏名 例 #define YES 1 main() { …….. } #undef YES #define YES 0 max() {…….. } YES原作用域 ❖宏定义可嵌套,不能递归 YES新作用域 例 #define MAX MAX+10 () ❖引号中的内容与宏名相同也不置换 例 #define PI 3.14159 printf(“2*PI=%f\n”,PI*2); 宏展开:printf(“2*PI=%f\n”,3.14159*2); ❖宏定义中使用必要的括号() 例 #define WIDTH 80 #define LENGTH WIDTH+40 var=LENGTH*2; 宏展开:var= 80+40 *2; ( ) ( ) 例 #define WIDTH 80 #define LENGTH WIDTH+40 var=LENGTH*2; 宏展开:var= 80+40 *2;
说明: ①宏名一般用大写字母。这不是语法规定,只是一种习惯。 ②可以使用宏名代替一个字符串 ③当宏定义在一行中写不下,需要在下一行继续时,只需 在最后一个字符后紧接一个反斜线“”。例如 #define LEaP year year %4==0 &&year%1001=0‖year%400=0 注意在第二行开始不要有空格,否则空格会一起被替换。 ④宏名的有效范围为定义命令之后到本源文件结束,除非 用# undef命令终止宏名的作用域。 ⑤对程序中用双引号括起来的字符串内的子串和用户标识 符中的成分不做替换 ⑥同一个宏名不能重复定义
说明: ① 宏名一般用大写字母。这不是语法规定,只是一种习惯。 ②可以使用宏名代替一个字符串 ③ 当宏定义在一行中写不下,需要在下一行继续时,只需 在最后一个字符后紧接一个反斜线“\” 。例如: #define LEAP_YEAR year % 4==0\ && year %100!=0 || year % 400==0 注意在第二行开始不要有空格,否则空格会一起被替换。 ④ 宏名的有效范围为定义命令之后到本源文件结束,除非 用#undef命令终止宏名的作用域。 ⑤ 对程序中用双引号括起来的字符串内的子串和用户标识 符中的成分不做替换。 ⑥ 同一个宏名不能重复定义
★带参数宏定义 令一般形式:# define宏名(参数表)宏体 例# define s(ab)a” 不能加空格 area=S(3, 2) 宏展开: area=3 *2 今宏展开:形参用实参换。其它字符保留 心宏体及各形参外一般应加括号() 例r# define s 相挺不弟以频表字容串“()Pr Z=POWER(X+y) 宏展开:z=X+y"x+ 般写成:# define POWer(×)(X)*(X) 宏展开:z=(X+y)*(Xy)
带参数宏定义 ❖一般形式: #define 宏名(参数表) 宏体 例 #define S (r) PI*r*r 相当于定义了不带参宏S,代表字符串“(r) PI*r*r” ❖宏展开:形参用实参换,其它字符保留 ❖宏体及各形参外一般应加括号() 例 #define S(a,b) a*b ……….. area=S(3,2); 宏展开: area=3*2; 不能加空格 例 #define POWER(x) x*x x=4; y=6; z=POWER(x+y); 宏展开:z=x+y*x+y; 一般写成: #define POWER(x) ((x)*(x)) 宏展开: z=((x+y)*(x+y));
例用宏定义和函数实现同样的功能 #define MAX(x, y)(x)>(y)?(x): (y) int max(int x, int y) i return(xy?x y) main i int a, b, c, d, t main i int a, b, c, d, t t=MAX(a+b, c+d) t=max(a+b, c+d) 宏展开:t=(a+b)>(c+o?(a+b)(c+d)
#define MAX(x,y) (x)>(y)?(x):(y) ……. main() { int a,b,c,d,t; ……. t=MAX(a+b,c+d); …… } 宏展开:t=(a+b)>(c+d)?(a+b):(c+d); int max(int x,int y) { return(x>y?x:y); } main() { int a,b,c,d,t; ……. t=max(a+b,c+d); ……… } 例 用宏定义和函数实现同样的功能
今带参的宏与函数区别 带参宏 函数 处理时间编译时 程序运行时 参数类型无类型问题 定义实参形参类型 处理过程不分配内存 分配内存 程序长度变长 不变 运行速度不占运行时间 调用和返回占时间
❖带参的宏与函数区别 带参宏 函数 处理过程 不分配内存 简单的字符置换 分配内存 先求实参值,再代入形参 处理时间 编译时 程序运行时 参数类型 无类型问题 定义实参,形参类型 程序长度 变长 不变 运行速度 不占运行时间 调用和返回占时间
§62文件包含 ★功能:一个源文◇直接换标准目录搜索 “”先在当前目录搜索,再搜索标准目录 包含进来 可指定路径 ★一般形式:# include“文件名” 或# nclude<文件名 ★处理过程:预编译时用被包含文件的内容取代 该预处理命令,再对“包含”后的文件作一个源 文件编译 # include“fle2.c file2.c A file2.c filel.c file 1
§6.2 文件包含 功能:一个源文件可将另一个源文件的内容全部 包含进来 一般形式: #include “文件名” 或 #include #include “file2.c” file1.c file2.c file1.c file2.c A B A 处理过程:预编译时,用被包含文件的内容取代 该预处理命令,再对“包含”后的文件作一个源 文件编译 <> 直接按标准目录搜索 “” 先在当前目录搜索,再搜索标准目录 可指定路径
★被包含文件内容 file3. c ☆源文件(*.C) file2.c 宏定义 ☆头文件(*h) filel.c 数据结构定义 A 函数说明等 ★文件包含可嵌套 # include“fie2c”|# include“fle3c” A B file3. c filel.c file2.c
被包含文件内容 ❖源文件(*.c) ❖头文件(*.h) 宏定义 数据结构定义 函数说明等 文件包含可嵌套 #include “file2.c” file1.c A file3.c C #include “file3.c” file2.c B file1.c A file3.c file2.c
powers h * 例文件包含举例 define sqr(x)(x)+(x) #define cube(x)((x)*(x)(x) #define quad()((x)*(x)*(x)*(x)) /*ch810c*/ #include #include"d: fengyilbkclpowers h #define maX poWer 10 void main i int n printf("number \t exp2 \t exp 3 t exp4n printf("---t- t-n"); for(n=l; n<=MAX POWER n++ printf( %2d t%3dt%4dt %5dn" n, sqr(n), cube(n), quad(n))
例 文件包含举例 /* powers.h */ #define sqr(x) ((x)*(x)) #define cube(x) ((x)*(x)*(x)) #define quad(x) ((x)*(x)*(x)*(x)) /*ch8_10.c*/ #include #include "d:\fengyi\bkc\powers.h" #define MAX_POWER 10 void main() { int n; printf("number\t exp2\t exp3\t exp4\n"); printf("----\t----\t-----\t------\n"); for(n=1;n<=MAX_POWER;n++) printf("%2d\t %3d\t %4d\t %5d\n",n,sqr(n),cube(n),quad(n)); }
63条件编译 “条件编译”:一般情况下,源程序中的所 有行都参加编译,但特殊情况下可能需要根据 不同的条件编译源程序中的不同部分,也就是 说对源程序的一部分内容给定一定的编译条件。 这种方式称作:条件编译
6.3 条件编译 “条件编译”:一般情况下,源程序中的所 有行都参加编译,但特殊情况下可能需要根据 不同的条件编译源程序中的不同部分,也就是 说对源程序的一部分内容给定一定的编译条件。 这种方式称作:“条件编译”