C程序设计
南京师范大学
地图学与地理信息系统 04级
专业选修课
主讲教师, 汪闽
第八章 编译预处理
8.1 宏定义
8.2 文件包含
8.3 条件编译
– 作用:对源程序编译之前做一些处理,生成扩展 C
源程序
?种类
? 宏定义 #define
? 文件包含 #include
? 条件编译 #if--#else--#endif等
– 格式:
?, #”开头
? 占单独书写行
? 语句尾不加分号
8.1.1 不带参数宏定义
? 一般形式,#define 宏名 [宏体 ]
? 功能,用指定标识符 (宏名 )代替字符序列 (宏体 )
宏展开:预编译时,用宏体替换宏名 ---不作语法检查
宏体可缺省,表示宏名
定义过或取消宏体
定义位置,任意 (一般在函数外面 )
作用域,从定义命令到文件结束
#undef可终止宏名作用域
格式,#undef 宏名
宏定义可嵌套,不能递归
例 #define MAX MAX+10 (?)
引号中的内容与宏名相同也不置换
宏定义中使用必要的括号 ()
8.1 宏定义
? 一般形式,#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));
8.1.2带参数宏定义
#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);
………
}
例 用宏定义和函数实现同样的功能
带参宏 函数
处理过程 不分配内存
简单的字符置换
分配内存
先求实参值,再代入形参
处理时间 编译时 程序运行时
参数类型 无类型问题 定义实参,形参类型
程序长度 变长 不变
运行速度 不占运行时间 调用和返回占时间
带参的宏与函数区别
– 功能:一个源文件可将另一个源文件的内容全部
包含进来
– 一般形式,#include,文件名,
或 #include <文件名 >
#include,file2.c”
file1.c
file2.c
file1.c
file2.c
A
B
A
处理过程:预编译时,用被包含文件的内容取代该预处理命
令,再对“包含”后的文件作一个源文件编译
<> 直接按标准目录搜索
“” 先在 当前目录 搜索,再搜索标准目录
可指定路径
8.2 文件包含
? 源文件 (*.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))
/*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.3 条件编译
条件编译可有效地提高程序的可移植性, 并广泛地应
用在商业软件中, 为一个程序提供各种不同的版本 。
8.3.1 #ifdef ~ #endif 和
#ifndef ~ #endif命令
1,一般格式
# ifdef 标识符
程序段 1;
[# else
程序段 2; ]
# endif
2,功能,当, 标识符, 已经被 #define命令定义过, 则
编译程序段 1,否则编译程序段 2。
( 1)在不同的系统中,一个 int 型数据占用的内存字节
数可能是不同的。
( 2)利用条件编译,还可使同一源程序即适合于调
试(进行程序跟踪、打印较多的状态或错误信息),又适
合高效执行要求。
3,关于 #ifndef~ #endif命令
格式与 #ifdef ~ #endif命令一样,功能正好与之相反。
8.3.2 #if ~ #endif
1,一般格式
# if 常量表达式
程序段 1;
[# else
程序段 2; ]
# endif
2,功能,当表达式为非 0(, 逻辑真, )
时,编译程序段 1,否则编译程序段 2。
[案例 8.2] 输入一个口令, 根据需要设置条件
编译, 使之能将口令原码输出, 或仅输出若干星
号, *” 。
/*案例代码文件名,AL8_2.C*/
#define PASSWORD 0 /*预置为输出星号 */
main()
{ ……
/*条件编译 */
#if PASSWORD /*源码输出 */
……
#else /*输出星号 */
……
#endif
……
}