13高级程序语言.ppt
第十三讲 编译预处理作用:对源程序编译之前做一些处理,生成扩展C源程序种类v宏定义#definev文件包含#includev条件编译#if-#else-#endif等格式:v“#”开头v占单独书写行v语句尾不加分号如 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”);1 宏定义不带参数宏定义v一般形式:#define 宏名 宏体v功能:用指定标识符(宏名)代替字符序列(宏体)v宏展开:预编译时,用宏体替换宏名-不作语法检查如#define YES 1#define NO 0#define PI 3.1415926#define OUT printf(“Hello,World”);宏体可缺省,表示宏名定义过或取消宏体v定义位置:任意(一般在函数外面)v作用域:从定义命令到文件结束v#undef可终止宏名作用域 格式:#undef 宏名例#define YES 1 main().#undef YES#define YES 0 max().YES原作用域YES新作用域v宏定义可嵌套,不能递归例#define MAX MAX+10 ()v引号中的内容与宏名相同也不置换例#define PI 3.14159 printf(“2*PI=%fn”,PI*2);宏展开:printf(“2*PI=%fn”,3.14159*2);v宏定义中使用必要的括号()例#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;带参数宏定义v一般形式:#define 宏名(参数表)宏体例#define S (r)PI*r*r相当于定义了不带参宏S,代表字符串“(r)PI*r*r”v宏展开:形参用实参换,其它字符保留v宏体及各形参外一般应加括号()例#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(xy?x:y);main()int a,b,c,d,t;.t=max(a+b,c+d);例 用宏定义和函数实现同样的功能v带参的宏与函数区别带参宏函数处理过程不分配内存简单的字符置换分配内存先求实参值,再代入形参处理时间编译时程序运行时参数类型无类型问题定义实参,形参类型程序长度变长不变运行速度不占运行时间调用和返回占时间连串运算符连串运算符#v 连串运算符连串运算符#可以运用在带参数的宏定义中可以运用在带参数的宏定义中v如果宏定义中宏表示的单词串中一个形参前面有如果宏定义中宏表示的单词串中一个形参前面有#,则在,则在进行宏替换时,对应的实参被转换成双引号括起来的字符进行宏替换时,对应的实参被转换成双引号括起来的字符串。串。v在实参前面和后面出现的空白符被忽略,实参中间出现的在实参前面和后面出现的空白符被忽略,实参中间出现的空白符被转换成字符串中的空白符。空白符被转换成字符串中的空白符。v例如:例如:#define PR(x)printf(#x)#define PR(x)printf(#x)PR(hello world!);PR(hello world!);将被替换成将被替换成 printf(“hello world!”);printf(“hello world!”);v如果实参中出现了转义字符,在替换时如果实参中出现了转义字符,在替换时#运算符会在其前运算符会在其前面加反斜线字符面加反斜线字符。例如:。例如:PR(“hello world!”):;PR(“hello world!”):;在预处理时将其替换成:在预处理时将其替换成:printf(“”hello world!”);printf(“”hello world!”);连结运算符连结运算符#v连结运算符连结运算符#通常出现在替换单词串中。通常出现在替换单词串中。v当进行宏替换时,在替换的单词串中出现的当进行宏替换时,在替换的单词串中出现的#将被删除,将被删除,在在#前面和后面出现的参数被相应的实参替换,并把它前面和后面出现的参数被相应的实参替换,并把它们连结成一个单词。们连结成一个单词。v如果连结后的单词构成宏名,则进一步进行宏替换。如果连结后的单词构成宏名,则进一步进行宏替换。v例如:例如:#define PRT(a)printf(“x”#a”=%dn”,x#a)PRT(1);经过预处理后,被替换成经过预处理后,被替换成printf(“x”1”=%dn”,x1);2 文件包含功能:一个源文件可将另一个源文件的内容全部包含进来一般形式:#include “文件名”或#include#include “file2.c”file1.cfile2.cfile1.cfile2.cABA处理过程:预编译时,用被包含文件的内容取代该预处理命令,再对“包含”后的文件作一个源文件编译 直接按标准目录搜索“”先在当前目录搜索,再搜索标准目录可指定路径被包含文件内容v源文件(*.c)v头文件(*.h)宏定义数据结构定义函数说明等文件包含可嵌套#include “file2.c”file1.cAfile3.cC#include “file3.c”file2.c Bfile1.cAfile3.cfile2.c例 文件包含举例/*powers.h */#define sqr(x)(x)*(x)#define cube(x)(x)*(x)*(x)#define quad(x)(x)*(x)*(x)*(x)#include#include d:bkcpowers.h#define MAX_POWER 10void main()int n;printf(numbert exp2t exp3t exp4n);printf(-t-t-t-n);for(n=1;n=MAX_POWER;n+)printf(%2dt%3dt%4dt%5dn,n,sqr(n),cube(n),quad(n);3条件编译 功能:根据外部条件决定只是编译程序中的某些部分,这样使得同一源程序在不同编译条件下可以编译不同的代码段,从而有利于程序的调试和移植。一般形式:第一种形式:#ifdef 标识符标识符 程序段程序段1#else 程序段程序段2#endif标识符已被#define命令定义过则对程序段1进行编译;否则对程序段2进行编译。例例7.7计算计算sin(x)=x-x3/3!+x5/5!-x7/7!+#include#include#define DEBUGvoid main(void)double s,t,x;int n;printf(“please input s:”);scanf(“%lf”,&x);t=x;n=1;s=x;do n=n+2;t=t*(-x*x)/(float)(n)-1)/(float)(n);s=s+t;#ifdef DEBUGprintf(“n=%d,t=%f,s=%fn”,n,t,s);#endifwhile(fabs(t)=1e-6;printf(“sin(%f)=%fn”,x,s);第二种形式:第二种形式:#ifndef 标识符标识符 程序段程序段1#else 程序段程序段2#endif标识符未被#define命令定义过则对程序段1进行编译;否则对程序段2进行编译。第三种形式:第三种形式:#if 常量表达式常量表达式 程序段程序段1#else 程序段程序段2#endif常量表达式的值为真(非0),则对程序段1 进行编译,否则对程序段2进行编译。例例7.8#include#define R 1void main(void)float c,r,s;printf(input a number:);scanf(%f,&c);#if R r=3.14159*c*c;printf(area of round is:%fn,r);#else s=c*c;printf(area of square is:%fn,s);#endif4 4 其他预处理命令其他预处理命令 行控制行控制#line#line 常数常数 “文件名文件名”作用作用:告诉编译程序,源程序的下一行的行号为命令行告诉编译程序,源程序的下一行的行号为命令行中的常数,当前正在处理的文件名称改为命令行中指中的常数,当前正在处理的文件名称改为命令行中指定的文件名。命令行中的文件名可以省略,这样原文定的文件名。命令行中的文件名可以省略,这样原文件名称不变。件名称不变。例如:例如:#line 30#line 30 “READREAD”则下一行的行号指定为则下一行的行号指定为3030,正在处理的文件改名为,正在处理的文件改名为READREAD。诊断控制诊断控制#error#error 字符序列字符序列此命令的作用是产生诊断信息,其中包含指定的字符此命令的作用是产生诊断信息,其中包含指定的字符序列。序列。