第 6章 编译预处理
作用,对源程序编译之前做一些处理,生成扩展 C
源程序
种类
宏定义 #define
文件包含 #include
条件编译 #if--#else--#endif等
格式:
,#”开头
占单独书写行
语句尾不加分号如 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 %100!=0 || year % 400==0
注意在第二行开始不要有空格,否则空格会一起被替换 。
④ 宏名的有效范围为定义命令之后到本源文件结束,除非用 #undef命令终止宏名的作用域 。
⑤ 对程序中用双引号括起来的字符串内的子串和用户标识符中的成分不做替换 。
⑥ 同一个宏名不能重复定义。
带参数宏定义
一般形式,#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)
…….
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);
………
}
例 用宏定义和函数实现同样的功能
带参的宏与函数区别带参宏 函数处理过程 不分配内存简单的字符置换分配内存先求实参值,再代入形参处理时间 编译时 程序运行时参数类型 无类型问题 定义实参,形参类型程序长度 变长 不变运行速度 不占运行时间 调用和返回占时间
§ 6.2 文件包含
功能:一个源文件可将另一个源文件的内容全部包含进来
一般形式,#include,文件名,
或 #include <文件名 >
#include,file2.c”
file1.c
file2.c
file1.c
file2.c
A
B
A
处理过程:预编译时,用被包含文件的内容取代该预处理命令,再对“包含”后的文件作一个源文件编译
<> 直接按标准目录搜索
“” 先在 当前目录 搜索,再搜索标准目录可指定路径
被包含文件内容
源文件 (*.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));
}
6.3 条件编译
,条件编译,,一般情况下,源程序中的所有行都参加编译,但特殊情况下可能需要根据不同的条件编译源程序中的不同部分,也就是说对源程序的一部分内容给定一定的编译条件 。
这种方式称作,“条件编译,。
条件编译命令主要包括以下几种形式:
1 #ifdef 标识符程序段 1
#else
程序段 2
#endif
它的含义是如果指定的标识符已经被 #define定义过,则只编译程序段 1,否则编译程序段 2。 其中 #else部分可以没有,即:
#ifdef 标识符程序段 1
#endif
它的含义是如果指定的标识符已经被 #define定义过,则编译程序段 1,否则跳过程序段 1。
2,#ifndef 标识符程序段 1
#else
程序段 2
#endif
含 义 是,如果 指 定的 标识 符 没有 被
#define定义过,则编译程序段 1,否则编译程序段 2。
3,#if 表达式程序段 1
#else
程序段 2
#endif
它的含义是,如果指定的表达式的值为真,则编译程序段 1,否则编译程序段 2。