第九章 预处理命令
作用,对源程序编译之前做一些处理,生成扩展 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”);
§ 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 (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);
………
}
例 用宏定义和函数实现同样的功能
< >
带参的宏与函数区别带参宏 函数处理过程 不分配内存简单的字符置换分配内存先求实参值,再代入形参处理时间 编译时 程序运行时参数类型 无类型问题 定义实参,形参类型程序长度 变长 不变运行速度 不占运行时间 调用和返回占时间
< >
§ 9.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
例 文件包含举例
( 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
#define S,%s”#include <format.h>
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);
}
§ 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
#endif
【 Example9.7】 输入一行字母,根据需要设置条件编译,使之能将字母全改为大写输出,或全改为小写字母输出。
#include <stdio.h>
#define LETTER 1
main()
{ char str[20]=“C Language”,c;
int i = 0;
while ((c=str[i])!=?\0?)
{i++;
#if LETTER
if (c >=?a? && c <=?z?)
c = c-32;
#else
if (c >=?A? && c <=?Z?)
c = c+32;
#endif
printf(“%c”,c);
}
}
作用,对源程序编译之前做一些处理,生成扩展 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”);
§ 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 (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);
………
}
例 用宏定义和函数实现同样的功能
< >
带参的宏与函数区别带参宏 函数处理过程 不分配内存简单的字符置换分配内存先求实参值,再代入形参处理时间 编译时 程序运行时参数类型 无类型问题 定义实参,形参类型程序长度 变长 不变运行速度 不占运行时间 调用和返回占时间
< >
§ 9.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
例 文件包含举例
( 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
#define S,%s”#include <format.h>
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);
}
§ 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
#endif
【 Example9.7】 输入一行字母,根据需要设置条件编译,使之能将字母全改为大写输出,或全改为小写字母输出。
#include <stdio.h>
#define LETTER 1
main()
{ char str[20]=“C Language”,c;
int i = 0;
while ((c=str[i])!=?\0?)
{i++;
#if LETTER
if (c >=?a? && c <=?z?)
c = c-32;
#else
if (c >=?A? && c <=?Z?)
c = c+32;
#endif
printf(“%c”,c);
}
}