第九章预处理命令 ★作用:对源程序编译之前做一些处理,生成扩展C 源程序 ★种类 今宏定义 define ◆文件包含# include ◆条件编译#f--#else-#endf等 ★格式 ◆“#开头 今占单独书写行 心语旬尾不加分号
第九章 预处理命令 作用:对源程序编译之前做一些处理,生成扩展C 源程序 种类 ❖宏定义 #define ❖文件包含 #include ❖条件编译 #if--#else--#endif等 格式: ❖“#”开头 ❖占单独书写行 ❖语句尾不加分号
§9.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”); §9.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” 不能加空格 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); ……… } 例 用宏定义和函数实现同样的功能
今带参的宏与函数区别 带参宏 函数 处理时间编译时 程序运行时 参数类型无类型问题 定义实参形参类型 处理过程不分配内存 分配内存 程序长度变长 不变 运行速度不占运行时间 调用和返回占时间
❖带参的宏与函数区别 带参宏 函数 处理过程 不分配内存 简单的字符置换 分配内存 先求实参值,再代入形参 处理时间 编译时 程序运行时 参数类型 无类型问题 定义实参,形参类型 程序长度 变长 不变 运行速度 不占运行时间 调用和返回占时间
§92文件包含 ★功能:一个源文◇直接换标准目录搜索 “”先在当前目录搜索,再搜索标准目录 包含进来 可指定路径 ★一般形式:# include“文件名” 或# nclude<文件名 ★处理过程:预编译时用被包含文件的内容取代 该预处理命令,再对“包含”后的文件作一个源 文件编译 # include“fle2.c file2.c A file2.c filel.c file 1
§9.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
(1)文件 format h 例文件包含举例 #define Pr printf define nl“r define d“9d #define did nl #define d2d d nl #define d3d dd nl #define D4D DD DNL #include efine s“%s main i int a.b.c.d char stringl=“ CHINA; a=l;b=2,c=3;d=4; PRODI, a PR(D2, a, b) PR(3, a, b, c) PR(D4, a, b, c, d) PR(S, String)
例 文件包含举例 (1)文件format.h #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 #include #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); }
§93条件编译 希望对程序中的一部分内容在满足一定条件时 编译,否则不编译,或编译另一部分内容。 # ifdef标识符若标识符已被# define定义 程序段1 过,则编译程序段1 #else 否则编译程序段2。 程序段2 #endif 或# indef标识符若标识符已被# define定义 程序段1过,则编译程序段1。 #endif
§9.3 条件编译 希望对程序中的一部分内容在满足一定条件时 编译,否则不编译,或编译另一部分内容。 一、 #ifdef 标识符 若标识符已被#define定义 程序段1 过,则编译程序段1, #else 否则编译程序段2。 程序段2 #endif 或 #ifdef 标识符 若标识符已被#define定义 程序段1 过,则编译程序段1。 #endif
、# ifndef标识符若标识符未被定义过, 程序段1则编译程序段1, #else 否则编译程序段2 程序段2 #endif 、#if表达式若表达式的值为真, 程序段1 则编译程序段1 #else 否则编译程序段2。 程序段2 fendi
二、 #ifndef 标识符 若标识符未被定义过, 程序段1 则编译程序段1, #else 否则编译程序段2。 程序段2 #endif 三、 #if 表达式 若表达式的值为真, 程序段1 则编译程序段1 #else 否则编译程序段2。 程序段2 #endif