第八章 编译预处理 西安工程大学
基本内容: 8.1概述 8.2宏定义 8.3文件包含 8.4条件编译
8.1概述 作用:对源程序编译之前做一些处理,生成扩展C源程序 格式 种类: “#”开头 占单独书写行 语句尾不加分号 宏定义 #define 文件包含 #include 条件编译 #if--#else--#endif等
8.2 宏定义 不带参数宏定义 一般形式: #define 宏名 [宏体] 功能:用指定标识符(宏名)代替字符序列(宏体) 宏体可缺省,表示宏名 定义过或取消宏体 8.2 宏定义 不带参数宏定义 一般形式: #define 宏名 [宏体] 功能:用指定标识符(宏名)代替字符序列(宏体) 定义位置:任意(一般在函数外面) 作用域:从定义命令到文件结束 #undef可终止宏名作用域 格式: #undef 宏名 如 #define YES 1 #define NO 0 #define PI 3.1415926 #define OUT printf(“Hello,World”); 例 #define YES 1 main() { …….. } #undef YES #define YES 0 max() {…….. YES原作用域 YES新作用域 例 #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; ( ) ( ) 宏展开:预编译时,用宏体替换宏名---不作语法检查 引号中的内容与宏名相同也不置换 如 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”); 例 #define PI 3.14159 printf(“2*PI=%f\n”,PI*2); 宏展开:printf(“2*PI=%f\n”,3.14159*2); 宏定义可嵌套,不能递归 宏定义中使用必要的括号() 例 #define MAX MAX+10 ()
带参数宏定义 一般形式: #define 宏名(参数表) 宏体 例 #define S(a,b) a*b ……….. 不能加空格 例 #define S(a,b) a*b ……….. area=S(3,2); 宏展开: area=3*2; 宏展开:形参用实参换,其它字符保留 宏体及各形参外一般应加括号() 例 #define S (r) PI*r*r 相当于定义了不带参宏S,代表字符串“(r) PI*r*r” 例 #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) ……. 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); { int a,b,c,d,t; t=max(a+b,c+d); ……… 例 用宏定义和函数实现同样的功能 < >
带参的宏与函数区别 带参宏 函数 处理过程 不分配内存 简单的字符置换 分配内存 先求实参值,再代入形参 处理时间 编译时 程序运行时 参数类型 无类型问题 定义实参,形参类型 程序长度 变长 不变 运行速度 不占运行时间 调用和返回占时间 < >
8.3 文件包含 功能:一个源文件可将另一个源文件的内容全部包含进来 一般形式: #include “文件名” 8.3 文件包含 功能:一个源文件可将另一个源文件的内容全部包含进来 一般形式: #include “文件名” 或 #include <文件名> <> 直接按标准目录搜索 “” 先在当前目录搜索,再搜索标准目录 可指定路径 处理过程:预编译时,用被包含文件的内容取代该预处理命令,再对“包含”后的文件作一个源文件编译 #include “file2.c” file1.c file2.c A B
被包含文件内容 文件包含可嵌套 源文件(*.c) 头文件(*.h) 宏定义 数据结构定义 函数说明等 #include “file2.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)) /*ch8_10.c*/ #include <stdio.h> #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)); } 例 文件包含举例
8.4条件编译 [Return] 条件编译可有效地提高程序的可移植性,并广泛地应用在商业软件中,为一个程序提供各种不同的版本。 8.4.1 #ifdef ~ #endif和#ifndef ~ #endif命令 8.4.2 #if ~ #endif [Return]
8.4.1 #ifdef ~ #endif和#ifndef ~ #endif命令 一般格式 #ifdef 标识符 程序段1; [#else 程序段2;] #endif 功能:当“标识符”已经被#define命令定义过,则编译程序段1,否则编译程序段2。 在不同的系统中,一个int 型数据占用的内存字节数可能是不同的。
[Return] 利用条件编译,还可使同一源程序即适合于调试(进行程序跟踪、打印较多的状态或错误信息),又适合高效执行要求。 关于#ifndef ~ #endif命令 格式与#ifdef ~ #endif命令一样,功能正好与之相反。 [Return]
8.4.2 #if ~ #endif 一般格式 #if 常量表达式 程序段1; [#else 程序段2;] #endif 功能:当表达式为非0(“逻辑真”)时,编译程序段1,否则编译程序段2。
[例8.2] 输入一个口令,根据需要设置条件编译,使之能将口令原码输出,或仅输出若干星号“*”。 /*例8.2代码文件名:AL8_2.C*/ #define PASSWORD 0 /*预置为输出星号*/ main() { …… /*条件编译*/ #if PASSWORD /*源码输出*/ …… #else /*输出星号*/ #endif }
[例8.3] 条件编译的使用 #define LETER 1 main() { char str[20]="C Language",c; int i=0; while((c=str[i])!='\0') { i++; #if LETER if(c>='a'&&c<='z') c=c-32; #else if(c>='A'&&c<='Z') c=c+32; #endif printf("%c",c); }