函数和编译预处理
c语言的预处理指令分3种 1宏定义 2条件编译 3文件包含
c语⾔的预处理指令分3种 1宏定义 2条件编译 3⽂件包含宏简介1.C语⾔在对源程序进⾏编译之前,会先对⼀些特殊的预处理指令作解释(⽐如之前使⽤的#include⽂件包含指令),产⽣⼀个新的源程序(这个过程称为编译预处理),之后再进⾏通常的编译所有的预处理指令都是以#开头,并且结尾不⽤分号2.预处理指令分3种 1> 宏定义 2> 条件编译 3> ⽂件包含3.预处理指令在代码翻译成0和1之前执⾏4.预处理的位置是随便写的5.预处理指令的作⽤域:从编写指令的那⼀⾏开始,⼀直到⽂件结尾,可以⽤#undef取消宏定义的作⽤6.宏名⼀般⽤⼤写或者以k开头,变量名⼀般⽤⼩写 宏定义可以分为2种:不带参数的宏定义和带参数的宏定义。
⼀、不带参数的宏定义1.⼀般形式#define 宏名字符串⽐如#define ABC 10右边的字符串也可以省略,⽐如#define ABC2.作⽤它的作⽤是在编译预处理时,将源程序中所有"宏名"替换成右边的"字符串",常⽤来定义常量.3.使⽤习惯与注意1> 宏名⼀般⽤⼤写字母,以便与变量名区别开来,但⽤⼩写也没有语法错误2> 对程序中⽤双引号扩起来的字符串内的字符,不进⾏宏的替换操作。
3> 在编译预处理⽤字符串替换宏名时,不作语法检查,只是简单的字符串替换。
只有在编译的时候才对已经展开宏名的源程序进⾏语法检查4> 宏名的有效范围是从定义位置到⽂件结束。
如果需要终⽌宏定义的作⽤域,可以⽤#undef命令5> 定义⼀个宏时可以引⽤已经定义的宏名#define R 3.0#define PI 3.14#define L 2*PI*R#define S PI*R*R举例1 #include <stdio.h>2#define COUNT 434int main()5 {6char *name = "COUNT";78 printf("%s\n", name);910int ages[COUNT] = {1, 2, 67, 89};1112#define kCount 41314for ( int i = 0; i<COUNT; i++) {15 printf("%d\n", ages[i]);16 }1718// 从这⾏开始,COUNT这个宏就失效19#undef COUNT2021//int a = COUNT 写这个报错2223return0;24 }⼆、带参数的宏定义1.⼀般形式#define 宏名(参数列表) 字符串2.作⽤在编译预处理时,将源程序中所有宏名替换成字符串,并且将字符串中的参数⽤宏名右边参数列表中的参数替换3.使⽤注意1> 宏名和参数列表之间不能有空格,否则空格后⾯的所有字符串都作为替换的字符串2> 带参数的宏在展开时,只作简单的字符和参数的替换,不进⾏任何计算操作。
江苏省计算机二级考试(vc)备考提纲(主要为基础知识) (1)
江苏省计算机二级考试备考提纲第一章Visual C++程序设计入门1>定义标识符规则:第一个字符只能是英文字母或下画线,后面可跟字母,数字,下画线;不能是C++语言的关键字。
2>键盘上除去3个字符:@,¥,其余的可显示字符在程序代码中均能使用。
3>编写程序的注意事项:/*和*/为多行注释,//为单行注释,从标识起至本行结束:程序一般包含输入输出编译预处理命令;C++的程序有且只有一个主函数main;对于C++编译器而言,一条语句可以写成若干行,一行内也可以写若干条语句,:而且它严格区分大小写字母。
4>运算符;1.求模/余运算符%,要求操作数必须都是整形数,若不是整型数必须将操作数强制转化成整型再进行求余运算,否则将出现编译错误,如(int)5.2%3=2; 2.若操作数中有负值,求余原则为:先取绝对值求余,余数取与被除数相同的符号,如-10%3=-1,10%-3=1. 3。
而除法运算符/和*,若两个操作数都是整型,则结果也是整型,若有一个是实型,则结果是实型。
4。
注意,如a=4,b=3,c=2,求a>b>c的值,因为原式=(a>b)>c,a>b值为1,则原式相当于1>c,所以最终结果是0。
5。
自增、减运算符的操作数不能是常量或表达式,如2++,(x+1)++都是不合法的,再如,2*a++等价于2*(a++). 6.条件运算符“?:”是C++中唯一的三目运算符,条件表达式的一般格式为:逻辑表达式1?表达式2:表达式3.,1为真执行2,为假执行3,注意:条件表达式的功能相当于条件语句,但一般不能取代if语句;表达式1,2,3类型可不同,此时条件表达式的值取较高的类型,如a>b?2:5.5,a<b时,值为5.5,a>b时,值为2.0,而不是2。
注意:逗号运算符计算方法:按先后顺序依次计算各个表达式的值,最后一个表达式的值作为整个逗号表达式的值。
函数与编译预处理实验报告
函数与编译预处理实验报告一、实验目的本次实验旨在通过编写程序,掌握函数与编译预处理的相关知识,了解函数的调用和返回机制以及编译预处理的作用。
二、实验环境操作系统:Windows 10开发工具:Code::Blocks 17.12编程语言:C语言三、实验内容1. 函数的定义和调用函数是C语言中的一个重要概念,它可以将一段代码封装成一个功能单元,方便代码重用和维护。
在本次实验中,我们需要掌握如何定义函数,并且了解函数的调用过程。
在C语言中,函数通常由以下几部分组成:返回类型函数名(参数列表){函数体;return 返回值;}其中,返回类型指定了函数返回值的类型;函数名是唯一标识符,用于调用该函数;参数列表指定了该函数需要传入的参数;函数体是一段具体的代码逻辑;return语句则将结果返回给调用者。
在调用一个函数时,我们需要按照以下格式进行:返回值变量 = 函数名(参数列表);其中,返回值变量接收该函数返回的结果;参数列表则按照定义顺序传入相应参数。
如果该函数不需要传入任何参数,则可以省略参数列表。
2. 函数指针除了直接调用一个已经定义好的函数外,我们还可以使用函数指针来间接调用一个函数。
函数指针和普通指针类似,它指向的是一个函数的地址。
通过函数指针,我们可以在运行时动态地确定需要调用哪个函数。
在C语言中,声明一个函数指针的方法如下:返回类型 (*指针变量名)(参数列表);其中,括号中的*表示这是一个指针变量;指针变量名则是该变量的标识符;参数列表和返回类型与被指向的函数相同。
通过以下代码可以将一个已经定义好的函数赋值给一个函数指针:int max(int a, int b){return a > b ? a : b;}int (*p)(int, int) = max;在上述代码中,p就是一个用于存储max函数地址的函数指针。
3. 编译预处理编译预处理是C语言中一项重要的功能,在编译过程中会对源代码进行一些预处理操作。
第7章 函数与预处理命令
例如:求两个数的最大值。 int max(int x,int y) { int z; 没有形式参数 z = x > y ? x : y; 为无参函数 return( z ); }
龙诚数码:
9
形参也可以这样定义 int max(x,y) int x,y; { int z; z = x > y ? x : y; return( z ); }
龙诚数码:
【例7.1】求一个整数的立方。
int cube (int x) /* 函数定义 */ { return (x * x * x); } main( ) 程序的执行总是 { int f, a; 从main函数开始 printf("\nEnter an integer number:"); scanf("%d", &a); 函数调用 f = cube (a); printf("%d * %d * %d = %d\n", a, a, a, f); 程序运行情况如下: } Enter an integer number:2 2*2*2=8
a=9,b=5
14
龙诚数码:
2.函数的类型与函数的返回值
15
【例7.6】输出两个数中的大数。 ⑴ 函数的类型 max(int x,int y) 说明: { int z; ①函数的类型 z=x>y?x:y; 决定了函数返 return (z); /* 返回z的值 */ 回值的类型。 } 若省略函数的 main( ) 类型,系统默 { int a,b,c; 认其为int型。 scanf("%d,%d",&a,&b); ②无返回值的 c=max(a,b); 函数应将其类 printf("max is %d\n",c); 型定义为void } (空)类型。
gcc编译过程的四个阶段
gcc编译过程的四个阶段1. 预处理(Preprocessing):预处理是编译过程的第一阶段。
预处理器负责对原始源文件进行处理,主要完成以下几个任务:-处理宏定义:预处理器会将源文件中的宏定义展开为相应的代码片段,并将其保存在一个临时文件中。
-处理条件编译指令:预处理器会根据条件编译指令的结果决定是否包含或排除一些代码片段。
- 处理#include指令:预处理器会将源文件中的#include指令所引用的其他文件插入到该指令所在的位置。
-移除注释:预处理器会删除源文件中的注释。
预处理后的文件成为扩展名为.i的中间文件,它包含了所有宏定义及展开后的代码。
编译是编译过程的第二阶段。
编译器将预处理生成的中间文件进行词法分析、语法分析和语义分析,生成相应的汇编代码。
主要过程如下:- 词法分析器将预处理生成的中间文件分解为一个个的词法单元(Token)。
- 语法分析器根据词法单元组织成的语法结构,生成抽象语法树(Abstract Syntax Tree,AST)。
-语义分析器对抽象语法树进行语义检查,包括类型检查和语义错误检查,确保程序的语义正确。
编译器将生成的汇编代码保存为扩展名为.s的汇编文件。
3. 汇编(Assembling):汇编是编译过程的第三阶段。
汇编器(Assembler)将编译器生成的汇编代码翻译成机器码,并生成目标文件。
具体过程如下:- 汇编器将汇编代码中的每一条汇编指令翻译成对应的机器码,同时为每个标号(Label)生成对应的地址。
-汇编器进行符号解析,将代码中引用的变量和函数与目标文件中的符号表进行匹配,生成正确的指令和地址。
汇编器将目标文件保存为扩展名为.o的目标文件。
4. 链接(Linking):链接是编译过程的最后阶段。
链接器(Linker)将目标文件与其他必要的库文件进行合并,生成最终的可执行文件或动态链接库。
主要过程如下:-链接器将目标文件中的函数和变量引用与其他目标文件中的定义进行匹配,解析外部引用,生成相应的引用表。
函数与预编译
4.1.3 函数的调用
函数调用:
所谓函数调用,就是使程序转去执行函数体。 在C++中,除了主函数外,其他任何函数都不能单独作 为程序运行。任何函数功能的实现都是通过被主函数直接或 间接调用进行的。
无参函数的调用格式: 函数名( ) 有参函数的调用格式: 函数名(实际参数表)
说明:
数据类型指函数返回值类型,可以是任一种数据类型, 默认为返回整型值(但新标准要求写明,不用默认方式)。 没有返回值应将返回值类型定义为void。
函数名采用合法标识符表示。 对无参函数,参数括号中的void通常省略,但括号不 能省略。 函数体由一系列语句组成。函数体可以为空,称为空 函数。
4.1.2 函数的定义
参数传递:
函数调用首先要进行参数传递,参数传递的方向是由实 参传递给形参。
传递过程是,先计算实参表达式的值,再将该值传递给 对应的形参变量。一般情况下,实参和形参的个数和排列顺 序应一一对应,并且对应参数应类型匹配(赋值兼容),即 实参的类型可以转化为形参类型。而对应参数的参数名则不 要求相同。
传值调用和引用调用:
假如不模块化
读多少行的程序能让你不头疼? main()当中能放多少行程序? 假如 cout()函数由10行代码替换, 那么你见过的程序会成什么样子?
如果所有代码都在main()当中,团队怎么合作?
•如果代码都在一个文件中,怎么团队合作?
模块化思想
• 模块各司其职
– 每个模块只负责一件事情,它可以更专心 – 便于进行单个模块的设计、开发、调试、测试和维护
4.1 函数的定义与调用 4.2 函数的参数传递,
c语言指令大全表解释
c语言指令大全表解释C语言指令是计算机科学领域中的基础知识,掌握好它们意味着能够进行高效的编码工作。
正因如此,这篇文章将会针对C语言指令大全表进行解释性阐述。
首先,我们需要了解表中不同的分类,包括:控制流、数据类型、函数库、编译指令和预处理指令。
这些分类代表了不同的操作方式和目的,我们需要根据需求选择不同的指令来使用。
接下来,让我们详细介绍一下C语言指令分类的不同之处和应用场景。
控制流指令:包括条件和循环等语句,它们是一些用来控制代码执行流程的代码片段。
例如:· if语句:用于条件判断,如果满足条件,则执行特定代码块。
· while循环语句:重复执行某一段代码,直到满足条件才退出。
· for循环语句:用于迭代一些操作,通常会有一个计数器来控制循环次数。
数据类型指令:C语言支持多种数据类型,我们需要根据实际需求来选择不同的数据类型。
例如:· int类型:用于表示整数值。
· float类型:用于表示浮点数值。
· char类型:用于表示单个字符。
· double类型:用于表示双精度浮点数。
函数库指令:函数库是预先编写好的一些代码片段,方便我们在程序中调用。
例如:· printf()函数:用于输出文字。
· scanf()函数:用于输入数据。
· pow()函数:用于数学运算,求一个数的n次方。
编译指令:用于告诉程序如何编译代码,之后我们可以在程序中使用它们。
例如:· #include指令:用于载入头文件。
· #define指令:用于定义宏。
预处理指令:是在编译代码之前执行的一系列操作。
例如:· #if指令:用于条件编译,根据条件判断是否编译。
· #ifdef指令:用于检查是否定义了某个宏。
在学习和理解这些指令的过程中,我们不仅需要了解每个指令的具体语法和用法,更要深入思考其潜在的含义和应用场景,学会如何灵活选择和使用它们。
C语言对源程序处理的四个步骤:预处理、编译、汇编、链接——预处理篇
C语⾔对源程序处理的四个步骤:预处理、编译、汇编、链接——预处理篇预处理1)预处理的基本概念C语⾔对源程序处理的四个步骤:预处理、编译、汇编、链接。
预处理是在程序源代码被编译之前,由预处理器(Preprocessor)对程序源代码进⾏的处理。
这个过程并不对程序的源代码语法进⾏解析,但它会把源代码分割或处理成为特定的符号为下⼀步的编译做准备⼯作。
2)预编译命令C编译器提供的预处理功能主要有以下四种:1)⽂件包含 #include2)宏定义 #define3)条件编译 #if #endif ..4)⼀些特殊作⽤的预定义宏a、⽂件包含处理1)⽂件包含处理⽂件包含处理”是指⼀个源⽂件可以将另外⼀个⽂件的全部内容包含进来。
C语⾔提供了#include命令⽤来实现“⽂件包含”的操作。
2)#include< > 与 #include ""的区别" "表⽰系统先在file1.c所在的当前⽬录找file1.h,如果找不到,再按系统指定的⽬录检索。
< >表⽰系统直接按系统指定的⽬录检索。
注意:1. #include <>常⽤于包含库函数的头⽂件2. #include " "常⽤于包含⾃定义的头⽂件 (⾃定义的头⽂件常⽤“ ”,因为使⽤< >时需要在系统⽬录检索中加⼊⾃定义头⽂件的绝对地址/相对地址否则⽆法检索到该⾃定义的头⽂件,编译时会报错)3. 理论上#include可以包含任意格式的⽂件(.c .h等) ,但我们⼀般⽤于头⽂件的包含。
b、宏定义1)基本概念在源程序中,允许⼀个标识符(宏名)来表⽰⼀个语⾔符号字符串⽤指定的符号代替指定的信息。
在C语⾔中,“宏”分为:⽆参数的宏和有参数的宏。
2)⽆参数的宏定义#define 宏名 字符串例: #define PI 3.141926在编译预处理时,将程序中在该语句以后出现的所有的PI都⽤3.1415926代替。
c语言 编译
c语言编译C语言是一种通用的高级编程语言,由美国计算机科学家丹尼斯·里奇于1972年在贝尔实验室开发。
C语言具有简洁、高效、可移植等特点,被广泛应用于系统软件、嵌入式软件、游戏开发、科学计算等领域。
C语言的编译过程是将源代码转换为可执行文件的过程,下文将详细介绍C语言的编译过程。
一、C语言的编译过程C语言的编译过程包括预处理、编译、汇编和链接四个阶段。
下面分别介绍这四个阶段的作用和实现方式。
1. 预处理预处理阶段是在编译之前进行的,其作用是将源代码中的预处理指令替换为实际的代码。
预处理指令以#号开头,包括#include、#define、#ifdef、#ifndef等指令。
预处理器将这些指令替换为实际的代码,生成一个新的源文件。
预处理后的源文件通常以.i作为扩展名。
2. 编译编译阶段是将预处理后的源代码转换为汇编代码的过程。
编译器将C语言源代码转换为一种称为中间代码的形式,中间代码是一种类似汇编语言的低级语言。
中间代码具有平台无关性,可以在不同的平台上进行优化和执行。
编译后的结果通常以.s作为扩展名。
3. 汇编汇编阶段是将编译生成的汇编代码转换为机器代码的过程。
汇编器将汇编代码转换为可执行的机器代码,并生成一个目标文件。
目标文件包括可执行代码、数据段、符号表等信息。
目标文件通常以.o 或.obj作为扩展名。
4. 链接链接阶段是将多个目标文件合并为一个可执行文件的过程。
链接器将目标文件中的符号和地址进行解析,生成一个可执行文件。
可执行文件包括操作系统可以直接执行的代码和数据,通常以.exe、.dll 或.so作为扩展名。
二、C语言编译器C语言编译器是将C语言源代码转换为可执行文件的工具,包括预处理器、编译器、汇编器和链接器四个部分。
C语言编译器可以在不同的平台上运行,生成可在目标平台上运行的可执行文件。
下面分别介绍常用的C语言编译器。
1. GCCGCC(GNU Compiler Collection)是一款开源的C语言编译器,由GNU组织开发。
C语言三种预处理功能
比如#define MAX(a,b) ((a)>(b)?(a):(b))则遇到 MAX(1+2,value)则会把它替换成: ((1+2)>(value)?(1+2):(value))注意事项和无参宏差不多。 但还是应注意
#define FUN(a) "a"
则,输入 FUN(345)会被替换成什么? 其实,如果这么写,无论宏的实参是什么,都不会影响其被替换成"a"的命运。也就是说, ""内的字符不被当成形参,即使它和一模一样。那么,你会问了,我要是想让这里输入 FUN(345)它就替换成"345"该怎么实现呢?请看下面关于#的用法
带参数
除了一般的字符串替换,还要做参数代换
格式: #define 宏名(参数表) 字符串
例如:
#define S(a,b) a*b area=S(3,2);//第一步被换为 area=a*b; ,第二步被换为 area=3*2;
(1)实参如果是表达式容易出问题
#define S(r) r*r area=S(a+b);//第一步换为 area=r*r;,第二步被换为 area=a+b*a+b;
值传递、返回值)。
冷门重点编辑
#define 用法
1、用无参宏定义一个简单的常量
#define LEN 12
这个是最常见的用法,但也会出错。比如下面几个知识点你会吗?可以看下:
(1)#define NAME "zhangyuncong" 程序中有"NAME"则,它会不会被替换呢? (2)#define 0x abcd 可以吗?也就是说,可不可以用不是标识符的字母替换成别的东 西? (3)#define NAME "zhang 这个可以吗? (4)#define NAME "zhangyuncong" 程序中有上面的宏定义,并且,程序里有句: NAMELIST 这样,会不会被替换成"zhangyuncong"LIST 四个题答案都是十分明确的。 第一个,""内的东西不会被宏替换。这一点应该大家都知道; 第二个,宏定义前面的那个必须是合法的用户标识符; 第三个,宏定义也不是说后面东西随便写,不能把字符串的两个""拆开; 第四个:只替换标识符,不替换别的东西。NAMELIST 整体是个标识符,而没有 NAME 标识符,所以不替换。 也就是说,这种情况下记住:#define 第一位置第二位置 (1) 不替换程序中字符串里的东西; (2) 第一位置只能是合法的标识符(可以是关键字); (3) 第二位置如果有字符串,必须把""配对; (4) 只替换与第一位置完全相同的标识符。 还有就是老生常谈的话:记住这是简单的替换而已,不要在中间计算结果,一定要替换出 表达式之后再算。
c语言编译的正确过程
c语言编译的正确过程以C语言编译的正确过程C语言是一种广泛应用于软件开发领域的高级编程语言,它具有语法简洁、执行效率高等特点,因此备受程序员的喜爱。
编译是将C 语言源代码转换为可执行文件的过程,本文将详细介绍C语言编译的正确过程。
1. 预处理在编译过程开始之前,C语言编译器会首先进行预处理。
预处理器会对源代码进行扫描,解析并处理以"#"开头的预处理指令,如宏定义、条件编译等。
预处理的主要目的是对源代码进行预处理,将所有预处理指令替换为实际代码。
2. 词法分析词法分析是编译器的下一个阶段。
在这个阶段,编译器会将源代码分解为一个个的词法单元,如关键字、标识符、运算符、常量等。
编译器会根据C语言的语法规则对每个词法单元进行解析和分类。
3. 语法分析语法分析是编译器的关键阶段之一。
在这个阶段,编译器会根据C 语言的语法规则,对词法单元进行语法分析,构建语法树。
语法树是一种树状结构,用于表示源代码的语法结构,便于后续的语义分析和代码生成。
4. 语义分析语义分析是编译器的另一个关键阶段。
在这个阶段,编译器会对语法树进行分析,检查语义错误和类型匹配等问题。
如果发现错误,编译器会生成相应的错误信息。
同时,编译器还会进行符号表的构建,用于记录变量、函数等符号的信息。
5. 代码生成代码生成是编译器的最后一个阶段。
在这个阶段,编译器会根据语义分析的结果,将语法树转换为目标平台的机器代码。
代码生成过程中,编译器会进行优化,以提高程序的执行效率。
优化包括常量折叠、循环展开、指令调度等技术。
6. 链接链接是将编译生成的目标文件与库文件进行合并,生成可执行文件的过程。
链接器会对目标文件进行符号解析和重定位,将各个模块的目标代码链接在一起,生成最终的可执行文件。
链接过程还包括对库文件的搜索和加载。
7. 运行编译生成可执行文件后,就可以运行程序了。
在运行过程中,操作系统会加载可执行文件到内存中,并按照指令进行执行。
C语言 函数 预处理
例6-1 文件包含预处理语句的使用
mymath.h的内容如下: # define ADD(x,y) (x+y) # define MUL(x,y) ((x)*(y)) # define PI 3.1415926 # define G 2.718 test.c内容如下: # include “mymath.h” main ( ) { int a=6,b=9; printf(“a+b=%d\n”,ADD(a,b)); printf(“a*b=%d\n”,MUL(a,b)); }
(5)使用宏次数多时,宏展开后源程序长,因为每 展开一次都使程序增长;而函数调用不使程序变长。 (6)宏替换不占用运行时间,而函数调用则占用运 6 行时间(分配内存单元、保护现场、值传递、执行 函数、返回函数值等)。
6.3 条件编译 .
在很多情况下,为了增强程序的可移植 性,C语言源程序中包含了各种版本的程序段, 但在某种情况下,只希望对一部分内容进行 编译,即为其指定编译的条件,此时称为条 件编译。
例6-3 带参数的宏定义 #define POWER(x) ((x)*(x)) main() { int y; scanf(“%d”,&y); printf(“y=%d, y*y=%d\n”,y, POWER(y)); }
从上述例题中可以看出,带参数的宏定义与函 数的使用非常相似,但二者是不同的。两者的区别 有如下几点: (1)函数调用时先求出实参的值,然后代入形参。 而带参数的宏定义只是进行简单的字符替换。 (2)函数调用是在程序运行时处理的,并分配临时 的内存单元。而宏展开则是在编译预处理时进行的, 在展开时并不分配内存单元,不进行值的传递,也 没有“返回值”。
第6章 编译预处理 章
C&&C++编译过程
C/C++编译过程C/C++编译过程主要分为4个过程1) 编译预处理2) 编译、优化阶段3) 汇编过程4) 链接程序一、编译预处理(1)宏定义指令,如#define Name TokenString,#undef等。
对于前一个伪指令,预编译所要做的是将程序中的所有Name用TokenString替换,但作为字符串常量的Name则不被替换。
对于后者,则将取消对某个宏的定义,使以后该串的出现不再被替换。
(2)条件编译指令,如#ifdef,#ifndef,#else,#elif,#endif等。
这些伪指令的引入使得程序员可以通过定义不同的宏来决定编译程序对哪些代码进行处理。
预编译程序将根据有关的文件,将那些不必要的代码过滤掉(3)头文件包含指令,如#include "FileName"或者#include <FileName>等。
在头文件中一般用伪指令#define定义了大量的宏(最常见的是字符常量),同时包含有各种外部符号的声明。
包含到c源程序中的头文件可以是系统提供的,这些头文件一般被放在/usr/include目录下。
在程序中#include它们要使用尖括号(< >)。
另外开发人员也可以定义自己的头文件,这些文件一般与c源程序放在同一目录下,此时在#include中要用双引号("")。
(4)特殊符号,预编译程序可以识别一些特殊的符号。
例如在源程序中出现的#line标识将被解释为当前行号(十进制数),上面程序实现了对宏line的运用(5)预处理模块预处理工作由#pragma命令完成,#Pragma命令将设定编译器的状态或者是指示编译器完成一些特定的动作。
#pragma指令对每个编译器给出了一个方法,在保持与C和C++语言完全兼容的情况下,给出主机或操作系统专有的特征。
依据定义,编译指示是机器或操作系统专有的,且对于每个编译器都是不同的。
程序编译的四个步骤
程序编译的四个步骤程序的编译过程通常分为四个步骤:预处理、编译、汇编和链接。
第一步:预处理(Preprocessing)预处理是编译过程的第一个步骤。
在这一步骤中,预处理器将对源代码进行处理,以便于后续的编译。
预处理器通常会执行以下任务:1.去除注释:将源代码中的注释(单行、多行注释)删除,以便于后续的处理。
2.展开宏定义:替换源代码中的宏定义,在源代码中使用宏定义的地方,将其替换为宏定义的内容。
3.处理条件编译指令:根据条件编译指令的条件,决定哪些代码需要编译,哪些代码需要忽略。
4.处理头文件包含指令:将头文件包含指令替换为头文件的内容,以确保源代码中可以使用头文件中定义的函数、变量等。
编译是预处理之后的一步,编译器将对预处理后的文件进行处理。
编译器通常会执行以下任务:1. 词法分析(Lexical Analysis):将源代码分解成一个个的词素,如关键字、标识符、运算符等,并生成相应的记号。
2. 语法分析(Syntax Analysis):根据词法分析生成的记号,将其按照一定的文法规则进行组织,构建抽象语法树。
3. 语义分析(Semantic Analysis):对抽象语法树进行分析,检查程序是否存在语义错误,如类型不匹配、未定义的变量等。
4. 代码生成(Code Generation):根据语义分析的结果,将抽象语法树转化为目标机器的汇编代码。
第三步:汇编(Assembly)汇编是编译过程的第三步,将编译器生成的汇编代码转化为机器码。
汇编器(Assembler)会执行以下任务:1.识别指令和操作数:根据汇编代码的语法规则,识别出每个指令以及对应的操作数。
2.生成机器码:将汇编指令和操作数翻译成机器码表示形式。
3.符号解析:解析并处理所有的符号引用,如函数、变量等的引用。
第四步:链接(Linking)链接是编译过程的最后一步,将编译器生成的目标代码和其他库文件进行合并。
1.解析外部符号引用:将目标代码中引用的外部符号(函数、变量等)与其他目标代码或库文件中的定义进行匹配。
C语言课件—编译预处理
#include <stdio.h> #define sqr(x) ((x)*(x))
#include "powers.h" #define cube(x) ((x)*(x)*(x))
void main() { int n调;试方法
#define quad(x) ((x)*(x)*(x)*(x))
print1f.("n编u辑mpboerw\teersx.ph2,保\t e存xp3\t exp4\n");
❖宏体及各形参外一般应加括号()
例 #define POWER(x) x*x
x=4; y=6;
z=POWER(x+y); 宏展开:z=x+y*x+y; 一般写成: #define POWER(x) 宏展开: z=((x+y)*(x+y));
((x)*(x))
Macro Definition
例. 带参数的宏与函数实现同样功能
第七章 编译预处理
概述 宏定义 文件包含 条件编译
Next chapter
Introduction
作用:编译程序的一部分,将特殊命令扩展到程 序中,生成扩展C源程序
种类
❖宏定义 #define ❖文件包含 #include ❖条件编译 #if--#else--#endif
格式:
❖“#”开头 ❖占单独书写行 ❖语句尾不加分号; ❖定义位置任意,决定其作用域
print2f.("-将---p\to-w---e\tr-s--.h--文\t-件---存--\n放"到); 某一目录下
for(n3=. 1;编n<辑=fMmAaXin;.nc,++将) powers.h包含进来
1、预处理——精选推荐
1、预处理1、预处理命令的定义 使⽤库函数之前,应该⽤#include引⼊对应的头⽂件。
这种以#号开头的命令称为预处理命令。
所谓预处理是指在进⾏编译时的第⼀遍扫描(词法扫描和语法分析)之前所做的⼯作。
预处理是C语⾔的⼀个重要功能,它由于处理程序负责完成。
当编译⼀个程序时,系统将⾃动调⽤预处理程序对程序中“#”开头的预处理部分进⾏处理,处理完毕之后可以进⼊源程序的编译阶段。
C语⾔源⽂件要经过编译、链接才能⽣成可执⾏程序: (1)编译(Compile)会将源⽂件(.c⽂件)转换为⽬标⽂件。
对于 VC/VS,⽬标⽂件后缀为.obj;对于GCC,⽬标⽂件后缀为.o。
编译是针对单个源⽂件的,⼀次编译操作只能编译⼀个源⽂件,如果程序中有多个源⽂件,就需要多次编译操作。
(2)链接(Link)是针对多个⽂件的,它会将编译⽣成的多个⽬标⽂件以及系统中的库、组件等合并成⼀个可执⾏程序。
在实际开发中,有时候在编译之前还需要对源⽂件进⾏简单的处理。
例如,我们希望⾃⼰的程序在 Windows 和 Linux 下都能够运⾏,那么就要在 Windows 下使⽤ VS 编译⼀遍,然后在 Linux 下使⽤ GCC 编译⼀遍。
但是现在有个问题,程序中要实现的某个功能在 VS 和GCC 下使⽤的函数不同(假设 VS 下使⽤ a(),GCC 下使⽤ b()),VS 下的函数在 GCC 下不能编译通过,GCC 下的函数在 VS 下也不能编译通过,怎么办呢? 这就需要在编译之前先对源⽂件进⾏处理:如果检测到是 VS,就保留 a() 删除 b();如果检测到是 GCC,就保留 b() 删除 a()。
这些在编译之前对源⽂件进⾏简单加⼯的过程,就称为预处理(即预先处理、提前处理)。
预处理主要是处理以#开头的命令,例如#include <stdio.h>等。
预处理命令要放在所有函数之外,⽽且⼀般都放在源⽂件的前⾯。
预处理是C语⾔的⼀个重要功能,由预处理程序完成。
c语言程序编译的流程
c语言程序编译的流程C语言是一种高级编程语言,它是一种通用的编程语言,可以用于开发各种类型的应用程序。
C语言程序编译的流程是指将C语言源代码转换为可执行文件的过程。
本文将详细介绍C语言程序编译的流程。
C语言程序编译的流程可以分为以下几个步骤:1. 预处理预处理是C语言程序编译的第一步。
在这个步骤中,编译器会对源代码进行一些预处理操作,例如宏替换、头文件包含等。
预处理器会将源代码中的宏定义替换为宏定义中的内容,并将头文件中的内容插入到源代码中。
预处理后的代码称为预处理文件。
2. 编译编译是C语言程序编译的第二步。
在这个步骤中,编译器会将预处理文件转换为汇编代码。
汇编代码是一种低级语言,它是机器语言的一种表现形式。
编译器会将C语言代码转换为汇编代码,这个过程称为编译。
3. 汇编汇编是C语言程序编译的第三步。
在这个步骤中,汇编器会将汇编代码转换为机器语言代码。
机器语言是计算机可以直接执行的语言,它是由0和1组成的二进制代码。
汇编器会将汇编代码转换为机器语言代码,这个过程称为汇编。
4. 链接链接是C语言程序编译的最后一步。
在这个步骤中,链接器会将机器语言代码和库文件链接在一起,生成可执行文件。
库文件是一些预编译的代码,它们可以被多个程序共享。
链接器会将程序中使用到的库文件链接到程序中,生成可执行文件。
以上就是C语言程序编译的流程。
下面我们将详细介绍每个步骤的具体内容。
1. 预处理预处理是C语言程序编译的第一步。
在这个步骤中,编译器会对源代码进行一些预处理操作,例如宏替换、头文件包含等。
预处理器会将源代码中的宏定义替换为宏定义中的内容,并将头文件中的内容插入到源代码中。
预处理后的代码称为预处理文件。
预处理器的工作原理是将源代码中的宏定义和头文件包含替换为实际的代码。
例如,下面是一个简单的宏定义:#define PI 3.1415926在预处理阶段,预处理器会将源代码中的所有PI替换为3.1415926。
这样,程序中所有使用到PI的地方都会被替换为3.1415926。
编译预处理的三种形式
编译预处理的三种形式编译预处理是指在编译阶段之前对源程序进行的一些处理,以便于编译器更好地理解和转换源程序。
这些处理包括宏定义、条件编译和文件包含等。
本文将分别介绍这三种形式的编译预处理。
一、宏定义宏定义是指用一个标识符来代表一段代码,然后在程序中使用该标识符来表示该段代码。
宏定义的语法如下:#define 标识符替换文本其中,标识符是一个由字母、数字和下划线组成的字符串,替换文本可以是任意合法的C语言代码。
1.简单宏定义简单宏定义是指只有一个替换文本的宏定义。
例如:#define PI 3.1415926这个宏定义将标识符PI替换为3.1415926。
在程序中使用该宏时,编译器会将所有的PI替换为3.1415926。
2.带参数的宏定义带参数的宏定义是指可以接受参数的宏定义。
例如:#define MAX(a, b) ((a) > (b) ? (a) : (b))这个宏定义可以接受两个参数a和b,并返回其中较大的值。
在程序中使用该宏时,需要传递两个参数,并且要用括号将每个参数括起来,以避免优先级问题。
3.带可变参数的宏定义带可变参数的宏定义是指可以接受可变数量参数的宏定义。
例如:#define PRINTF(format, ...) printf(format, ##__VA_ARGS__)这个宏定义可以接受一个格式化字符串和可变数量的参数,并将其传递给printf函数。
在程序中使用该宏时,需要传递至少一个参数,格式化字符串中使用%来表示要输出的数据类型,可变参数用逗号分隔。
二、条件编译条件编译是指根据不同的条件选择性地编译某些代码。
条件编译通常用于实现跨平台或调试功能。
1.#ifdef和#ifndef#ifdef和#ifndef分别表示“如果定义了某个宏”和“如果未定义某个宏”。
例如:#ifdef DEBUGprintf("debug mode\n");#endif这段代码只有在DEBUG宏已经被定义时才会被编译。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
函数和编译预处理(第五章)一、单项选择题1.关于函数,以下正确的描述是( B)A. 函数的定义可以嵌套,但函数的调用不可以嵌套B. 函数的定义不可以嵌套,但函数的调用可以嵌套C. 函数的定义可以嵌套,函数的调用也可以嵌套D. 函数的定义和函数的调用都不可以嵌套2.关键字inline用于定义内联函数,定义时,是将该关键字( D )A. 取代定义函数的类型标识符B. 取代被定义的函数名C. 加在类型标识符之后D. 加在类型标识符之前3.以下不正确的描述为( B )A. 在函数之外定义的变量称为外部变量,外部变量是全局变量。
B. 在函数之内说明的外部变量是局部变量,仅在本函数中有效。
C. 在函数之外说明的静态变量也是全局变量,其作用域是文件作用域。
D. 文件作用的域范围是一个程序文件,但必须符合“定义在前,使用在后”的规则。
4.以下正确的描述为( C )A. 每个C++程序必须在开头用预处理命令#include <iostream.h>B. 预处理命令必须位于C++源程序的首部C. 在C++中,预处理命令都以#开头D. C++语言的预处理命令只能实现宏定义和条件编译的功能5.在下面的函数原型说明中,存在着语法错误的是( D )A.void BC(int a,int);B.void BD(int , int);C.void BE(int , int=5);D.int BF(int x ; int y);6.下列哪个不是重载函数在调用时选择的依据( C )A. 形参类型B. 形参个数C. 函数返回值类型D. 函数名7.在一个源程序文件中有以下函数定义,其中( D )是重载函数。
A.ABC B. BCDC. ACDD. 全部A) int sum(float x,int y) B) float sum(float x,int y,char z){...} {...}C) float sum(float x,float y) D) int sum(int x,int y,char z){...}{...}8.有一个函数原型abc(float x,float y);该函数的返回值类型为( C )A. voidB. doubleC. intD. float9.在程序中,函数声明语句正确位置是( D )A. 随便任何位置B. 不包含在另一函数中的任何位置。
C. 该函数使用前的任何位置D. 该函数使用前的任何位置,但不包含在另一函数中10.C++构造内联函数的思想是( A )A. 用空间换时间B. 用时间换空间C. 用形参换实参D. 用实参换形参11.在以下存储类型中,( D )是用于定义动态类型的变量。
A. static 和autoB. register 和externC. register和 staticD. auto 和register12.下列各类变量,哪个不是局部变量( B )A. register型变量B. 外部static变量C. auto型变量D. 函数形参13.以下关于调用函数时,形、实参结合的通用规则不正确的描述为(B )A. 实参可以是变量,也可以是常数或者表达式。
B. 实参的个数可以多于形参,也可以少于形参。
C. 系统将为形参分配内存单元。
D. 实参必须与对应的形参类型相兼容。
14.假设有宏定义:#define NUM 15#define DNUM NUM+NUM则表达式DNUM/2+NUM*2的值为( C )A. 45B. 67C. 52D. 9015.对于一个功能不太复杂,并且要求加快执行速度,选用(A )合适。
A. 内联函数B. 重载函数C. 递归调用D. 嵌套调用16.若有以下函数调用语句:fun((a+b,(x,y)),fun(n+k,d),(a,b));其中实参的个数是(A )A.3 B. 4 C. 5 D. 617.以下叙述下不正确的是(D )A. 在函数中,通过return返回函数值。
B.函数中可以有多条return语句。
C.主函数main()也可以带有形参。
D.调用函数,必须在一条独立的语句中完成。
18.设有函数定义:int f1(void){return 100,200;},调用函数f1()时,(C )A. 函数返回值为100B. 函数返回二个值100和200C. 函数返回值为200D. 语句“return 100,200;”语法错,不能调用函数19.调用宏定义和语句:#define M(a,b) a*b; //Eint x=M(3+4,5+6),y; //Fy=M(3,4); //G则(B )A. 编译时,E行有语法错B. 编译时,F行有语法错C. 编译时,G行有语法错D. 编译时,F行和G行有语法错20.在一个源文件中定义的全局变量的作用域为(D )A. 本文件的全部范围B. 本程序的全部范围C. 本函数的全部范围D. 从定义该变量的位置开始至本文件的结束21.对于下面几个函数,(C )是重载函数。
void f(int x) { ... } //1int f(int y) { ... } //2int f(int i , int j ) { ... } //3float k(int x ) { ... } //4A. 4个全部B. 1和4C. 2和3D. 3和422.编译系统根据同一个程序中(A )识别重载函数。
A. 形、实参的个数和类型的差别B. 形、实参的个数和函数名的差别C. 形、实参个数,类型和返回值类型的差别D. 形、实参的类型和返回值类型的差别23. 以下正确的说法是(D )A. 用户若需要调用标准库函数,调用前必须重新定义B. 用户可以直接调用所有标准库函数C. 用户可以定义和标准库函数重名的函数,但是在使用时调用的是系统库函数D. 用户可以通过文件包含命令将系统库函数包含到用户源文件中,然后调用系统库函数24. 以下叙述中不正确的是(D )A. 在不同的函数中可以使用相同名字的变量B. 函数中的形式参数是局部变量C. 在一个函数内定义的变量只在本函数范围内有效D. 在一个函数内的复合语句中定义的变量在本函数范围内有效二、填空题1.当在块作用域内的局部变量与全局变量同名时,则局部变量优先。
2.在类型前加关键字inline 定义的函数称为内联函数。
3.有如下宏定义#define X(a) (a)*a,则表达式X(4+5)的值为41 。
4.变量分为全局和局部两种,全局变量没有赋初值时,其缺省值为0。
5.内联函数的实质是在编译时把函数的函数体直接插入到函数调用处。
6.C++在进行宏扩展时,不作做任何语法检查和计算,只对宏名进行简单替换。
7.在函数的递归调用时,若在A函数内调用B函数,而在B函数内调用A函数,称为间接递归。
8.若myheadfile.h为用户自定义的头文件,则在程序开始处包含此头文件的格式为#include”myheadfile.h”。
9.具有块作用域的变量都是局部变量。
10.递归函数在执行过程中,存在二个过程是递归和回推。
11.有如下宏定义#define X(a) a*a,则表达式X(4+5)的值为29 。
12.在for语句中说明的循环控制变量具有的作用域是包含for语句的内层块。
13.静态整数变量有确定的初值,其缺省的初值为 0 。
14.在设计递归方法程序时,为防止进入无穷递归,在递归程序中一定要有递归结束条件。
15.若在程序前定义了#define ring(x,y,z) x*y/z ,且在程序中有语句s=ring(25,6,1.5); 则程序运行时执行的相应语句(宏扩展)为s=25*6/1.5。
16. 在一个函数定义的函数体中又出现直接或间接地调用该函数本身称为函数的递归调用。
17.若要定义只能用于本文件的内部函数时,则在函数的类型标识符前加修饰词static 。
18. 函数原型为abc(float x,char y);该函数的返回值类型为 int 。
19.在C++程序中,当函数调用在前、函数定义在后时,则应在主调函数中,在调用前增加对被调函数的原型说明。
20.在函数的递归调用时,若在A函数内调用A函数,称为直接递归。
21. 若有宏定义:#defin X 2#defin Y(n)((X+1)*n)则执行语句:z=2*(X+Y(X+2));后,z的值是 20 。
三、阅读程序题1.#include <iostream.h>void test(int x, int y=20){cout<<”x=”<<x<<”;y=”<<y<<endl;}void main( ){int x=10;int y=15;test(x,y);test(x);}执行以上程序后,输出的第一行结果:x=10;y=15第二行结果:x=10;y=202.#include<iostream.h>#define ABCvoid main(){float s, x, y, z;s=1;y=z=0;while(s<=10){cin>>x;y+=x;s++;}z=y/10;#ifdef ABCcout<< y<<endl;#endifcout<<z<<endl;}问题1:程序中二个输出语句分别输出什么信息。
问题2:取消或保留语句#define ABC,对程序输出有何影响?(1) y输出10个实数之和z输出10个实数的平均值(2) 取消语句#define ABC,则程序输出为10个实数的平均值。
保留语句#define ABC,则程序输出为10个实数之和及10个实数的平均值。
3.# include <iostream.h># define AREA(a,b) a*b# define PREIMETER(a,b) 2*(a+b)void main(void){ cout<<"Area="<<AREA(1+2,3+4)<<endl;cout<<"Preimete="<<PREIMETER(5,6)<<endl;}以上程序段执行后,输出是:Area=11Preimete=224.#include <iostream.h>int i=2,j=3;int f(int a,int b){ int c=0; static int d=3;d++;if(a>b) c=1;else if (a==b) c=0;else c=-1;i=j+1;return (c+d);}void main(){ int p;p=f(i,j);cout<<i<<’,’<<j<<’,’<<p<<endl;i=i+1; p=f(i,j);cout<<i<<’,’<<j<<’,’<<p<<endl;i=i+1; p=f(i,j);cout<<i<<’,’<<j<<’,’<<p<<endl;}执行以上程序后,第一行输出:4,3,3第二行输出:4,3,6第三行输出:4,3,75.# include <iostream.h>int a=5;void main() {int a=10,b=20;cout <<a<<'\t'<<b<<endl;{ int a=0,b=0;for (int i=1; i<6; i++) {a+=i; b+=a;}cout <<a<<'\t'<<b<<'\t'<<::a<<endl;}cout <<a<<'\t'<<b<<endl;}执行以上程序后,第一行输出:10,20第二行输出:15,35,5第三行输出:10,206.#include <iostream.h>void fun( );void main( ){int i;for(i=0;i<5;i++)fun();}void fun( ){static int m=0;cout<<++m;}执行以上程序时,输出结果是:123457.#include <iostream.h>void main(void){ int i;void add1(void);void add2(void);for (i=0;i<3;i++){ add1();add2();cout<<endl;}}void add1(void){ int x=0;x++;cout<<x<<'\t';}void add2(void){ static int x=0;x++;cout<<x<<'\t';}执行以上程序时,输出结果是: 1 11 21 38.#include <iostream.h>int fac(){int b=0;static int c=3;b++;c++;return b+c;}void main( ){for(int j=0;j<3;j++) cout<<fac()<<’\t’; }执行如上程序后,输出结果是:5 6 79.#include <iostream.h>int fun(double a,double b);void main( ){cout<<fun(2.5,3.5)<<endl;}int fun(double a,double b){return a*b;}执行以上程序时,输出结果是:810.#include<iostream.h>void fun(int n){cout<<n%10;if(n<10) return;else fun(n/10);}void main(){int n;cin>>n;fun(n);cout<<endl;}执行以上程序时输入:469257,输出结果是:752964 11.#include <iostream.h>void change(int x,int y){x+=10;y+=10;cout<<”x=”<<x<<”;y=”<<y<<endl;}void main( ){int x=10;int y=15;cout<<”x=”<<x<<”;y=”<<y<<endl;change(x,y);cout<<”x=”<<x<<”;y=”<<y<<endl;}上面的程序编译并运行,程序的输出结果有三行,请写出程序运行的第一行结果:x=10;y=15第二行结果:x=20;y=25第三行结果:x=10;y=1512.#include<iostream.h>int f(int x){ static int u=1;x+=x;return u*=x;}void main(void){ int x=10;cout<<f(x)<<'\n';cout<<f(x)<<'\n';}程序输出的第一行是:20程序输出的第二行是:40013.#include <iostream.h>#define M 20#define N 10#define L(r) r*rvoid main( ){cout<<L(M)<<endl;cout<<L(N+N)<<endl;}以上程序段执行后,输出是 40012014.#include <iostream.h>int i=1;int fac(int n){ static int f=1;f=f*n;return(f);}void main(){ int i,p=0;for(i=1;i<=3;i++)p=fac(i);cout<<p<<endl;p=0;for(i=1;i<=3;i++)p=p+::i++;cout<<p<<',';p=1;for(i=1;i<=3;i++)p=p*--::i;cout<<p<<endl;}程序输出的第一行是:6程序输出的第二行是:6, 6程序中的::i是指:全局变量i15.//文件名:cppfile_1.cpp#include <iostream.h>int x=1,y=2;static int z=3;extern void add(void);void main( ){ add();cout<<"x="<<x<<'\t'<<"y="<<y <<'\t'<<"z="<<z<<endl;}//文件名:cppfile_2.cpp#include <iostream.h>extern int x,y;void add(void){ x+=3;y+=4;cout<<"x="<<x<<'\t'<<"y="<<y<<endl;}执行以上程序时,输出结果是:x=4 y=6x=4 y=6 z=316.#include <iostream.h>int f(int a,int b){ if(a>b)return b+a; else b*f(++a,--b);return a*f(a,b);}如果主函数中有语句 cout<<f(1,3)<<'\n';输出是:24如果主函数中有语句 cout<<f(3,1)<<'\n';输出是:417.#include <iostream.h>int i=2,j=3;int f(int a,int b){ int c=0;static int d=3;d++;if(a>b) c=1;else if (a==b) c=0;else c=-1;i=j+1;return (c+d);}void main(){ int p;p=f(i,j);cout<<i<<’,’<<j<<’,’<<p<<endl;i=i+1; p=f(i,j);cout<<i<<’,’<<j<<’,’<<p<<endl;i=i+1; p=f(i,j);cout<<i<<’,’<<j<<’,’<<p<<endl;}执行该程序的第一行输出是:4,3,3第二行输出是:4,3,6第三行输出是:4,3,718. #include<iostream.h>fun3(int x){static int a=3;a=x;return(a);}void main(){int k=2,m=1,n;n=fun3(k); n=fun3(m+n);cout<<n<<endl;}程序输出是:319. #include <iostream.h>int i=2,j=3;int f(int a,int b){ int c=0, d=3;d++;if(a>b) c=1;else if (a==b) c=0;else c=-1;i=j+1;return (c+d);}void main(){ int p;p=f(i,j);cout<<i<<’,’<<j<<’,’<<p<<endl;i=i+1; p=f(i,j);cout<<i<<’,’<<j<<’,’<<p<<endl;i=i+1; p=f(i,j);cout<<i<<’,’<<j<<’,’<<p<<endl;}执行该程序的第一行输出是:4,3,3第二行输出是:4,3,5第三行输出是:4,3,520. 若有宏定义如下:#define X 5#define Y X+1#define Z Y*X+2则执行以下语句后,int a; a=Y;cout<<Z;cout<<--a<<endl;输出结果是:12521. #include <iostream.h>int fac(int n){ int z;if (n>0)z=n*fac(n-2);elsez=1;return z;}void main(void){ int x=7,y;y=fac(x);cout<<y<<endl;}执行以上程序时,输出结果是:10522. #include <iostream.h>void fun( );int n=5;void main( ){int n=10;cout<<++n<<endl;fun();}void fun( ){cout<<n++<<endl;}执行以上程序时,输出结果是:11 523. # include <iostream.h>void SB(char ch) {switch(ch){case 'A': case 'a':cout <<"well!"; break;case 'B': case 'b':cout <<"good!"; break;case 'C': case 'c':cout <<"pass!"; break;default:cout <<"bad!"; break;}}void main() {char a1='b',a2='C',a3='f';SB(a1);SB(a2);SB(a3);SB('A');cout <<endl;}执行以上程序时,输出结果是:good! pass! bad! well!24. 设有宏定义如下:#define MIN(x,y) (x)>(y)?(x):(y)#define T(x,y,r) x*r*y/4则执行以下语句后,int a=1,b=3,c=5,s1,s2;s1=MIN(a=b,b-a);s2=T(a++,a*++b,a+b+c);s1的值为:3s2的值为:2725. #include <iostream.h>int add(int x,int y=8);void main( ){int a=4;cout<<add(a)<<endl;c out<<add(a,add(a))<<endl;}int add(int x,int y){return x+y;}执行以上程序时,输出结果是:12 1626. #include <iostream.h>fun(int x){ static int a=3; a=x; return(a); }void main(){ int k=2,m=1,n; n=fun(k); n=fun(m+n); cout<<n<<endl; } 执行以上程序时,输出结果是:3四、完善程序题1.此程序为打印如下图形,请将程序补充完整。