第八章编译预处理 所谓编译预处理是指,在对源程序进行编译之前, 先对源程序中的编译预处理命令进行处理;然后再将处 理的结果,和源程序一起进行编译,以得到目标代码。 8.1宏定义 8.2文件包含
8.1 宏定义 8.2 文件包含 第八章 编译预处理 所谓编译预处理是指,在对源程序进行编译之前, 先对源程序中的编译预处理命令进行处理;然后再将处 理的结果,和源程序一起进行编译,以得到目标代码
★作用:对源程序编译之前做一些处理生成扩展C 源程序 ★种类 今宏定义# define 今文件包含# include 令条件编译井f-#else-#endi等 ★格式: “#”开头 今占单独书写行 语句尾不加分号
作用:对源程序编译之前做一些处理,生成扩展C 源程序 种类 ❖宏定义 #define ❖文件包含 #include ❖条件编译 #if--#else--#endif等 格式: ❖“#”开头 ❖占单独书写行 ❖语句尾不加分号
8.1宏定义 ★不带参数宏定义 令一般形式:# define宏名[宏体] 令功能用指定标识符(宏名代替字待序列(宏体) 什思YE图数外Ⅲ) 作用域歉仑义冷到文件结束 并ndee条t方作域5926 格武 finetune则rt( Hello, World)); 例# define v|DTH80 宏展升# ine LENGTH WIDTH+40p S原作用域 va=LENGTH*2 中宏展开:=80+402; S新作用域 宏开:pn(?=%i,:141534
如 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”); 8.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宏名(参数表)宏体 例# define s(ab)a”b 不能加空格 area=S(3 宏展开: area=3*2 宏展开:形参用实参换,其它字符保留 宏体及各形参外一般应加括号() 例r#enS 相9不果以乐感表字容串“(0)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) i int a, b, c, d, t; mainO 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); ……… } 例 用宏定义和函数实现同样的功能
今带参的宏与函数区别 带参宏 函数 处理时间编译时 程序运行时 参数类型|无类型问题 定义实参,形参类型 处理过程不分配内存 分配内存 简单的字符置换先求实参值再代入形参 程序长度变长 不变 运行速度不占运行时间 调用和返回占时间
❖带参的宏与函数区别 带参宏 函数 处理过程 不分配内存 简单的字符置换 分配内存 先求实参值,再代入形参 处理时间 编译时 程序运行时 参数类型 无类型问题 定义实参,形参类型 程序长度 变长 不变 运行速度 不占运行时间 调用和返回占时间
8.2文件包含 ★功能:一个源文(直接按标准目录搜索 部 含进來 先在当前目录搜索,再搜索标准目录 可指定路径 ★一般形式:# include文件名” 或# include 处理过程:预编译时,用被包含文件的内容取代该预处理命 令,再对“包含”后的文件作一个源文件编译 #include"file2. c"KH file2.c B A file2.c filel.c filel.c
8.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“fle2.c”1# include“file3.c” 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*(x)) /*ch810c* #include #" powers. h #define maX power 10 void maino i int n printf number t exp 2 \t exp 3 t exp4n") printf("----t----t-----t------In for(n=l n=MAX POWER; n++) printf("%2d t %3d\t%4dt %5d\n", n, sqr(n), cube(n), quad(n)) Powers h Ch8 10c
例 文件包含举例 /* 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 "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)); }