《第1章C/C语言概述(修改).ppt》由会员分享,可在线阅读,更多相关《第1章C/C语言概述(修改).ppt(38页珍藏版)》请在三一办公上搜索。
1、第1章C/C+语言概述,1.1 从C到C+的程序结构 1.2 程序书写规范1.3 数据类型1.4 运算符和表达式1.5 基本语句 1.6 函数1.7 指针和引用1.8 作用域和存储类型 1.9 预处理,1.1从C到C+的程序结构,C+程序是由预处理命令、语句、函数、变量(对象)、输入与输出以及注释等几个基本部分组成的。代码中的main表示主函数,每一个C+程序都必须包含一个且只能包含一个main函数。main函数体是用一对花括号“”和“”括起来的,函数体中包括若干条语句,每一条语句都以分号“;”作为结束的标志。C+用标准输入输出的头文件iostream.h替代了C语言的stdio.h,用cin
2、、cout和操作运算符、等实现并扩展了C语言的scanf和printf函数功能。,1.2程序书写规范,标识符命名 标识符是用来标识变量名、函数名、数组名、类名、对象名、类型名、文件名等的有效字符序列。标识符命名需要遵守其合法性、有效性和易读性的原则。(1)合法性C+规定标识符由大小写字母、数字字符(09)和下划线组成,且第一个字符必须为字母或下划线。任何标识符中都不能有空格、标点符号、运算符及其他非法字符。标识符的大小写是有区别的,并且不能和系统的关键字同名。(2)有效性 标识符的长度最好不要超过32个,有的编译系统只能识别前32个字符,前32个字符相同的两个不同标识符被有的系统认为是同一个标
3、识符。(3)易读性 做到“见名知意”就可以达到易读性的目的。许多程序员采用“匈牙利标记法”来定义标识符。这种方法是:在每个变量名前面加上表示数据类型的小写字符,变量名中每个单词的首字母均大写。,1.2程序书写规范,缩进和注释 程序在书写时不要将程序的每一行都由第一列开始,应在语句前面加进一些空格,称为“缩进”,或是在适当的地方加进一些空行,以提高程序的可读性。每个花括号占一行,并与使用花括号的语句对齐。花括号内的语句采用缩进书写格式,缩进量为四个字符(一个默认的制表符)。注释是为了提高程序的可读性。注释本身对编译和运行并不起作用。在程序中,凡是放在“/*.*/”之间或以“/”开头行尾的内容都是
4、注释的内容,其中,/*.*/注释方式可以出现在程序中的任何位置。一般来说,注释应在编程的过程中进行,且注释内容一般有:源程序的总体注释、函数注释及其他的少量注释。一般不要陈述那些一目了然的内容,以免影响注释的效果。,1.3数据类型,1.3.1基本数据类型 C/C+的基本数据类型有字符型(char)、整型(int)和浮点型(float、double)三种。这些基本数据类型还可用short、long、signed和unsigned来修饰。需要注意的是:(1)C+可以有布尔型(bool),在计算机内,编译系统将true表示成整数1,false表示成整数0,因此也可把布尔型看成是一个整型。(2)无符号
5、(unsigned)和有符号(signed)的区别在于数值最高位的含义。有符号类型最高位是符号位,其余各位表示数值大小;无符号类型的各个位都用来表示数值大小;相同基本数据类型的signed和unsigned的数值范围是不同。例如,无符号字符型值的范围为0255,有符号字符型值的范围为-128-127。(3)char、short、int和long可统称为整型。默认时,char、short、int和long本身是有符号(signed)的。1.3.2常量整型常量 整型常量可以用十进制、八进制和十六进制来表示。十进制整型常量即十进制整数;八进制整型常量是以0开头的数;十六进制整型常量是以0 x或0X开
6、头的数,它由0至9、A至F或a至f组成。,1.3数据类型,注意:(1)整型常量中的长整型(long)要以L或小写字母l作为结尾。(2)整型常量中的无符号型(unsigned)要以U或u作为结尾。浮点型常量 浮点型常量即实数,它有十进制数或指数两种表示形式。十进制数形式是由整数部分和小数部分组成的(注意必须有小数点)。指数形式采用科学表示法,它能表示出很大或很小的浮点数。注意字母E(或e)前必须有数字,且后面的指数必须是整数。浮点型常量以F(或f)结尾的表示单精度类型(float),以L(或小写字母l)结尾的,表示长双精度类型(long double)。若浮点型常量没有任何说明,表示双精度类型(
7、double)。字符常量字符常量是用单引号括起来的一个字符。C/C+可以用一个“”开头的字符来表示特殊含义的字符常量。例如n,代表一个换行符,而不是表示字母n。这种将反斜杠()后面的字符转换成另外意义的方法称为转义表示法,n称为转义字符。,1.3数据类型,字符串常量 C+可以使用字符串常量。字符串常量是一对双引号括起来的字符序列。例如:“Hello,World!n”。字符串常量中还可以包含空格、转义字符或其他字符。并且必须在同一行书写,若一行写不下,则需要用来连接。字符常量和字符串常量主要的区别:(1)字符常量用单引号括起来的,仅占一个字节;字符串常量是用双引号括起来的,至少占用两个字节。“a
8、”是字符串常量,它的长度不是1而是2,除字符a之外,末尾还有个0字符。每个字符串的末尾都有一个这样的字符。(2)字符常量实际上是整型常量的特殊形式,它可以参与常用的算术运算;而字符串常量则不能。符号常量在C+中,除了用C语言的#define定义符号常量外,还常常用const来定义符号常量。,1.3数据类型,1.3.3变量变量的定义定义变量是用下面的格式:;(1)可以将同类型的变量定义在一行语句中,不过变量名要用逗号(,)分隔。但在同一个程序块中,不能有两个相同的变量名。(2)注意在C+中没有字符串变量类型,字符串是用字符类型的数组或指针来定义的。(3)与C语言相比,C+变量的定义比较自由。变量
9、的初始化 程序中常需要对一些变量预先设置初值,这一过程称为初始化。在C/C+中,可以在定义变量时同时使变量初始化。C+变量的初始化还有另外一种形式,它与C语言不同。例如:int nX(1),nY(3);表示nX和nY是整型变量,它们的初值分别为1和3。,1.3数据类型,1.3.4数据类型转换 自动转换 自动转换是将数据类型从低到高的顺序进行转换强制类型转换 强制类型转换是在程序中通过指定数据类型来改变图所示的类型转换顺序,将一个变量从其定义的类型改变为另一种新的类型。强制类型转换有两种格式:()()这里的“类型名”是任何合法的C/C+数据类型,通过类型的强制转换可以将“表达式”转换成指定的类型
10、。,1.3数据类型,1.3.5数组 数组的定义定义一个数组可按下列格式进行:.后面的常量表达式用于确定数组的维数和大小。一般表示某维大小的常量表达式中不能包含变量,但可以包括常量和符号常量,其值必须是一个确定的整型数值,且数值大于1。数组元素的引用 引用格式:.数组的赋值数组中的元素既可以在数组定义的同时赋初值,也可以在定义后赋值。对于二维或多维数组需要说明的是:(1)初始化数组的值的个数不能多于数组元素个数,初始化数组的值也不能通过跳过逗号的方式来省略。(2)对于二维数组来说,如果对全部元素都赋初值,则定义数组时对第一维的大小可以忽略,但第二维的大小不能省。(3)只对部分元素赋初值,可有两种
11、说明方式:一种是以“行”为单位,依次列出部分元素的值;另一种是以数组元素的排列顺序依次列出前面部分元素的值。字符数组 C/C+语言中,一个字符串是用一个以空字符0作为结束符的字符串来表示的。,1.3数据类型,1.3.6结构体定义结构体 结构体定义的格式为:struct;.;结构变量名列表;结构体定义是以关键字struct作为标志的,应是一个有效的标识符。在结构体中的每个成员都必须通过“成员定义”来确定成员名及其类型。注意:结构体的成员变量类型既可以是基本数据类型,也可以是其他合法的类型。例如:struct STUDENTPERSONone;/用已定义的结构体类型声明成员float eng,ph
12、y,math,poli;/英语、物理、数学和政治的成绩;,1.3数据类型,1.3.6结构体结构体变量的初始化和引用 结构体变量的初始化的一般形式是在变量后面加上:=;例如:struct POINT int x;int y;spot=20,40;/依次使spot中的x为20,y为40或POINT pt1=100,200;/使pt1中的x为100,y为200POINT pt2=100;/只是使pt2中的x为100当一个结构体变量定义之后,就可引用这个变量。使用时,遵循下列规则:(1)只能引用结构体变量中的成员变量,使用格式:.。(2)若成员本身又是一个结构体变量,引用时需要用多个成员运算符一级一级
13、地找到要访问的成员。,1.3数据类型,1.3.7共用体共用体和结构体最大的区别是:共用体在任一时刻只有一个成员处于活动状态,且共用体变量所占的内存长度等于各个成员中最长成员的长度,而结构体变量所占的内存长度等于各个成员的长度之和。共用体中各个成员所占内存的字节数各不相同,但都是从同一地址开始的。这种多个成员变量共用一个内存区的技术,能有效地降低程序所占的内存空间。定义一个共用体可用下列格式:union;.;共用体变量名列表;/注意最后的分号不要忘记。共用体除了关键字(union)不同外,其使用方法均与结构体相同。,1.3数据类型,1.3.8枚举类型 枚举类型是一系列有标识符的整型常量的集合,增
14、加程序代码的可读性。格式:enum 枚举变量;/注意最后的分号不要忘记。enum是关键字,枚举常量表中的枚举常量名之间要用逗号分隔。默认时,系统为每一个枚举常量都对应一个整数,并从0开始,逐个增1。例如:enum Colors Black,Blue,Green=4,Cyan,Red=8,Yellow,White one;则各枚举常量对应的整数依次为0,1,4,5,8,9,10。枚举变量最终的值只能等于该枚举类型中的某个枚举常量,而不能用一个整型数值直接赋值,并且不要在定义枚举类型的同时,再对枚举常量、枚举变量及枚举类型名重新定义。1.3.9用typedef定义类型使用关键字typedef可以将
15、已有的类型名用新的类型名来代替,格式:typedef;,1.4运算符和表达式,1.4.1算术运算符 算术运算符包括双目的加减乘除四则运算符、求余运算符以及单目的正负运算符。算术运算符如下所示:+(正号运算符,如+4,+1.23等)-(负号运算符,如-4,-1.23等)*(乘法运算符,如6*8,1.4*3.56等)/(除法运算符,如6/8,1.4/3.56等)%(模运算符或求余运算符,如40%11等)+(加法运算符,如6+8,1.4+3.56等)-(减法运算符,如6-8,1.4-3.56等)注意:(1)两个整数相除,将保留整数部分,不是四舍五入;进行浮点数除法,结果是浮点型。如7/5.0、7.0
16、/5、7.0/5.0的结果都是1.4。(2)求余运算要求参与运算的两个操作数都是整型,其结果是两个数相除的余数。例如40%5的结果是0,40%11的结果是7。要理解负值的求余运算,例如40%-11结果是7,-40%11结果是-7,-40%-11结果也是-7。,1.4运算符和表达式,1.4.2赋值运算符复合赋值 10种复合赋值运算符:+=,-=,*=,/=,%=,赋值符的结合性是从右至左的,上述的赋值是这样的过程:首先对赋值表达式nNum3=100求值,即将100赋值给nNum3,同时该赋值表达式取得值100;然后将该值赋给nNum2,这是第二个赋值表达式,该赋值表达式也取得值100;最后将10
17、0赋给nNum1。,1.4运算符和表达式,1.4.3关系运算符“关系运算”是比较两个操作数是否符合给定的条件。符合条件,则关系表达式的值为“真”,否则为“假”。在C+编译系统中,往往将“真”表示为“true”或1,将“假”表示为“false”或0。而任何不为0的数被认为是“真”,0被认为是“假”。关系运算需要两个操作数,所以关系运算符都是双目运算符。C+提供了下列6种关系运算符:,=,=(相等于),!=(不等于)前4种的优先级相同且高于后面的两种。关系运算符的优先级低于算术运算符 1.4.4逻辑运算符 用于将多个关系表达式或逻辑量(“真”或“假”)组成一个逻辑表达式。提供3种逻辑运算符:!逻辑
18、非(单目)将“真”变“假”,“假”变“真”。&逻辑与(双目)当两个操作数都是“真”时,结果才为“真”,否则为“假”。|逻辑或(双目)当两个操作数中有一个是“真”时,结果就为“真”,而只有当它们都为“假”时,结果才为“假”。“逻辑非”、“逻辑与”和“逻辑或”的优先级依次从高到低,“逻辑非”的优先级比算术运算符和关系运算符高,“逻辑与”和“逻辑或”的优先级比关系运算符要低。,1.4运算符和表达式,1.4.5位运算符 对操作数按其在计算机内表示的二制数逐位地进行逻辑运算或移位运算,参与运算的操作数只能是整型常量或变量。提供了六种位运算符:(按位求反,单目运算符)将一个二进制数的每一位求反。(右移,双
19、目运算符)将左操作数的二进制值向右移动指定的位数。右移后,移出的低位舍弃。如果是无符号数则高位补0;如果是有符号数,则高位补符号位或补0,不同的编译系统对此有不同的处理方法。&(按位与,双目运算符)将两个操作数对应的每个二进制位分别进行逻辑与操作。(按位异或,双目运算符)将两个操作数对应的每个二进制位分别进行异或操作。|(按位或,双目运算符)将两个操作数对应的每个二进制位分别进行逻辑或操作。1.4.6三目运算符 唯一的三目运算符是条件运算符,格式:?:。注意:只有在表达式2后面才能出现分号结束符,“表达式1”和“表达式2”中都不能有分号。,1.4运算符和表达式,1.4.7增1和减1运算符+和-
20、可放在变量的左边也可在变量的右边,分别称前缀运算符和后缀运算符。注意:若前缀运算符和后缀运算符仅用于某个变量的增1和减1,则这两个都是等价的,但如果将这两个运算符和其他运算符组合在一起,在求值次序上就会产生根本的不同:(1)如果用前缀运算符对一个变量增1(减1),则在将该变量增1(减1)后,用新的值在表达式中进行其他的运算。(2)如果用后缀运算符对一个变量增1(减1),则用该变量的原值在表达式进行其他的运算后,再将该变量增1(减1)。1.4.8逗号运算符 逗号运算符是优先级最低的运算符,它可以使多个表达式放在一行上。计算时,从左至右逐个计算每个表达式,最终整个表达式的结果是最后计算的那个表达式
21、的类型和值。例如:j=(i=12,i+8);式中,i=12,i+8 是含逗号运算符的表达式,计算次序是先计算表达式i=12,然后再计算i+8,整个表达式的值是最后一个表达式的值,即i+8的值20,从而j的结果是20。,1.4运算符和表达式,1.4.9 sizeof运算符 sizeof的目的是返回操作数所占的内存空间大小(字节数),具有两种格式:sizeof()sizeof()同一类型的操作数在不同的计算机中占用的存储字节数可能不同,因此sizeof的结果有可能不一样。例如sizeof(int)的值可能是4,也可能是2。1.4.10 new和delete 运算符new返回指定类型的一个指针,分配
22、失败则返回0。系统自动根据double类型的空间大小开辟一个内存单元,并将地址放在指针p中。运算符delete操作是释放new请求到的内存。需要注意的是:(1)运算符delete必须用于先前new分配的有效指针。如果使用了未定义的其它任何类型的指针,就会带来严重问题。(2)用new也可指定分配的内存大小。(3)new可以为数组分配内存,当释放时,必须告诉delete数组有多少个元素。,1.5 基本语句,1.5.1表达式语句、空语句和复合语句 表达式语句、空语句及复合语句是一些系统顺序执行的语句又称为顺序语句。任何一个表达式加上分号就是一个表达式语句。如果表达式是一个空表达式,那么构成的语句称为
23、空语句。空语句仅为语法的需要而设置,并不执行任何动作。复合语句是由两条或两条以上的语句组成的,并由一对花括号()括起来的语句。又称块语句。复合语句中的语句可以是单条语句(包括空语句),也可以再包含复合语句。注意:在复合语句中定义的变量只作用于该复合语句的范围,而在复合语句外,这些变量却不能被调用。1.5.2 选择语句 条件语句 格式:if()else 注意:(1)条件语句中的表达式一般为逻辑表达式或关系表达式。表达式的类型也可以是任意的数值类型(包括整型、浮点型、字符型等)。(2)适当添加一些花括号(“”)来增加程序的可读性。(3)条件语句中的语句1和语句2也可是if条件语句,这就形成了if语
24、句的嵌套。else总是和其前面最近的if配套的。,1.5基本语句,开关语句 开关语句switch能很好地解决嵌套太多这种问题,它具有下列形式:switch()case:语句1case:语句2.case:语句n default:语句n+1 当表达式的值与case中某个表达式的值相等时,就执行该case中“:”号后面的所有语句。若case中所有表达式的值都不等于表达式的值,则执行default:后面的语句,若default不存在,则跳出switch 结构。注意:(1)switch后面的表达式可以是整型、字符型或枚举型的表达式,而case后面的常量表达式的类型必须与其匹配。(2)多个case可以共有
25、一组执行语句。(3)若同一个case后面的语句是复合语句,则这些语句可以不用花括号(“”)括起来。(4)case语句起标号作用,因此每一个case常量表达式的值必须互不相同,否则会出现编译错误。(5)合理使用break语句使其跳出switch 结构,以保证结果的正确性;若没有break语句,则后面的语句继续执行,直到switch 结构的最后一个花括号(“”)为止才跳出该结构。,1.5基本语句,1.5.3 循环语句while循环语句语句格式:while()是循环体,可以是一条语句,也可以是复合语句。当表达式为非0时便开始执行while循环体中的语句,然后反复执行,每次执行都会判断表达式是否为非0
26、,若等于0,则终止循环。注意:循环体中应有使循环趋向结束的语句。do.while循环语句语句格式:do while()当语句执行到while时,将判断表达式是否为非0值,若是,则继续执行循环体,直到下一次表达式等于0为止。,1.5基本语句,for循环语句 语句格式:for(表达式1;表达式2;表达式3)一般情况下,表达式1用作循环变量的初始化,表达式2是循环体的判断条件,当等于非0时,开始执行循环体,再计算表达式3,再判断表达式2的值是否为非0,若是,再执行循环体,再计算表达式3,如此反复,直到表达式2等于0为止。注意:(1)表达式1、2、3都可以省略,分号“;”不能省略。省略表达式1,不影响
27、正确执行,但所需要的一些变量及其相关的数值要在for语句之前定义。省略表达式2,则表达式2的值被认为是“真”,循环无终止地进行下去,应在循环体中使用break语句。省略表达式3,应在设计循环结构时保证表达式2的值有等于0的可能,以便能终止循环。(2)表达式1和表达3可以是一个简单的表达式,也可以是逗号表达式,即包含两个或两个以上的简单表达式,中间用逗号分隔。甚至还可以定义变量。(3)由于循环体是由任何类型的语句组成的,因此在循环体内还可以包含前面的几种循环语句,这样就形成了循环的嵌套。1.5.4 break、continue语句 跳出循环结构或重新开始循环,就得使用break和continue
28、语句,格式如下:break;/可以从一个循环体跳出,也可以跳出switch结构。continue;/用于依靠条件判断而进行循环的循环语句,如for、while语句。对于for语句来说,continue的目的是将流程转到for语句的表达2和表达式3。,1.6函数,1.6.1函数的定义和调用 函数的定义格式 定义的格式如下:()函数名后面必须跟一对圆括号“()”以区别变量名及其他用户定义的标识名。形参写在括号内,参数可以是0,一个或多个,多个参数间要用逗号分隔。函数的函数体由在一对花括号中的若干条语句组成,用于实现这个函数执行的动作。不允许在一个函数体中再定义函数。函数的声明 声明函数格式:();
29、形参的变量名可以省略。但注意,函数声明的内容应和函数的定义应相同。函数的调用 函数调用格式为:();实参与形参相对应,是实际调用函数时所给定的常量、变量或表达式,必须有确定的值。,1.6函数,1.6.2带默认形参值的函数注意:(1)当函数既有声明又有定义后,不能在函数定义中指定默认参数。(2)默认参数值可以是全局变量、全局常量、一个函数。不可以是局部变量,因为默认参数的函数调用是在编译时确定的,局部变量的值在编译时无法确定。(3)一个函数中有多个默认参数时,则形参分布中,默认参数应从右到左逐渐定义。函数调用时,系统按从左到右的顺序将实参与形参结合,当实参的数目不足时,系统将按同样的顺序用声明或
30、定义中的默认值来补齐所缺少的参数。1.6.3函数的递归调用 C+允许在调用一个函数的过程中出现直接地或间接地调用函数本身,这种情况称为函数的“递归”调用,相应的函数称为递归函数。递归调用编写的程序简洁清晰,但每次调用函数时,都需要分配内存来保存现场和返回地址,内存空间开销很大,有时会引起栈内存溢出。,1.6函数,1.6.4内联函数 程序的执行过程中,调用函数时先要保存主调函数的现场和返回地址,然后程序转移到被调函数的起始地址继续执行。被调函数执行结束后,先恢复主调函数的现场,取出返回地址并将返回值赋给函数调用本身,最后在返回地址处开始继续执行。当函数体比较小时,且执行的功能比较简单时,这种函数
31、调用方式的系统开销相对较大。内联函数把函数体的代码直接插入到调用处,将调用函数的方式改为顺序执行直接插入的程序代码,这样可以减少程序的执行时间,但同时需要更多的内存空间。内联函数的定义方法是在函数定义时,在函数的类型前增加关键字inline。注意:(1)内联函数也要遵循定义在前,调用在后的原则。(2)需要定义成的内联函数不能含有循环、switch和复杂嵌套的if语句。(3)递归函数是不能被用来做内联函数的。编译器是否将用户定义成的内联函数作为真正的内联函数处理由它自行决定。,1.6函数,1.6.5函数的重载 它允许多个同名的函数存在,但同名的各个函数的形参必须有区别:形参的个数不同,或者形参的
32、个数相同,但参数类型有所不同。需要说明的是:(1)重载函数必须具有不同的参数个数或不同的参数类型,若只有返回值的类型不同是不行的。(2)当函数的重载带有默认参数时,应该注意避免二义性。例如:int fun(int a,int b=0);int fun(int a);是错误的。因为如果有函数调用fun(2)时,编译器无法准确地确定应调用哪个函数。,1.7指针和引用,1.7.1指针和指针变量指针变量格式定义:*,*,.;“*”是一个定义指针变量的说明符,指针变量前面都需这样的“*”标明。例如:int*pInt1,*pInt2;/pInt1,pInt2是指向整型变量的指针1.7.2&和*运算符&(取
33、地址运算符)、*(取值运算符)运算符“&”只能对变量操作,作用是取该变量的地址。运算符“*”用于指针类型的变量操作,作用是取该指针所指内存单元中存储的内容。说明:(1)在使用指针变量前,一定要对其进行初始化或使其有确定的地址数值。(2)指针变量只能赋以一个指针的值,若给指针变量赋了一个变量的值而不是该变量的地址或者赋了一个常量的值,则系统会以这个值作为地址。根据这个“地址”读写的结果将是致命的。(3)两个指针变量进行赋值,必须使这两个指针变量类型相同。(4)给指针变量赋值实际上是“间接”给指针所指向的变量赋值。,1.7指针和引用,1.7.3指针和数组数组中所有元素都是依次存储在内存单元中的,每
34、个元素都有相应的地址。数组名代表数组中第一个元素的地址,即数组的首地址。指针变量和数组的数组名在本质上是一样,因此指向数组的指针变量实际上也可像数组变量那样使用下标,而数组变量又可像指针变量那样使用指针。例如:pii与*(pi+i)及ai是等价的,*(a+i)与*(pi+i)是等价的。1.7.4指针和结构体 指针也可指向结构体类型变量。若将结构体变量看成一个整体,那么指向结构体变量数组的指针操作和指向数组的指针操作是一样的。例如若有:PERSONmany10,*pp;pp=many;/等价于pp=则pp+i与many+i是等价的,(pp+i)-name与manyi.name是等价的,等等。,1
35、.7指针和引用,1.7.5函数的指针传递 如果函数的某个参数是指针,对这函数的调用就是按地址传递的,由于函数形参指针和实参指针指向同一个地址,因此形参内容的改变必将影响实参。1.7.6引用引用是给一个已定义的变量起一个别名,系统不为引用类型变量分配内存空间,只是使引用类型变量与其相关联的变量使用同一个内存空间。引用类型变量定义格式:/错误:不能建立数组的引用(4)可以引用一个结构体。引用本身不是一种数据类型,所以没有引用的引用,也没有引用的指针。,1.7指针和引用,1.7.7函数的引用传递 指针作为函数的参数时,形参改变后相应的实参也会改变。但如果以引用作为参数,则既可以实现指针所带来的功能,
36、而且更加简便自然。一个函数能使用引用传递的方式是在函数定义时将形参前加上引用运算符“时,实际上是将实参a、b的地址存放到系统为形参分配的内存空间中,形参的任何操作都会改变相应的实参的数值。引用除了可作为函数的参数外,还可作为函数的返回值。,1.8作用域和存储类型,1.8.1作用域 块作用域块就是块语句。在块中声明的标识符,其作用域从声明处开始,一直到结束块的花括号为止。具有块作用域的标识符称作局部标识符,块作用域也称作局部作用域。说明:(1)当标识符的作用域完全相同时,不允许出现相同的标识符名。而当标识符具有不同的作用域时,允许标识符同名。(2)Visual C+中,在for语句中声明的标识符
37、,其作用域是包含for语句的那个内层块,而不是仅仅作用于for语句,这与标准C+不一样。函数原型作用域 在声明函数原型所指定的参数标识符的作用范围。这个作用范围是在函数原型声明中的左、右括号之间。在函数原型中声明的标识符可以与函数定义中说明的标识符名称不同。所声明的标识符与该函数的定义及调用无关,可以在函数原型声明中只作参数的类型声明,而省略参数名。,1.8作用域和存储类型,函数作用域 只有goto语句中的标号标识符具有函数作用域。具有函数作用域的标识符在声明它的函数内随处可见,但在此函数之外不可见。由于goto语句的滥用导致程序流程无规则、可读性差。因此现代程序设计方法不主张使用goto语句
38、。文件作用域 在函数外定义的标识符或用extern说明的标识符称为全局标识符。全局标识符的作用域称为文件作用域,它从声明之处开始,直到文件结束一直是可见的。在同一个作用域中,不能对同名的标识符作多种不同的声明。而当块作用域内的标识符与全局标识符同名时,局部标识符优先,且在块作用域内使用作用域运算符“:”来引用与局部标识符同名的全局标识符。,1.8作用域和存储类型,1.8.2变量的存储类型 规定了变量的生存期。无论是全局变量还是局部变量,编译系统往往根据其存储方式定义、分配和释放相应的内存空间。全局变量是指其作用域是文件作用域的变量,局部变量是指其作用域小于文件作用域的变量。自动类型(auto)
39、一般用自动存储类型声明的变量都是限制在某个程序范围内使用的。从系统角度来说,自动存储类型变量是采用堆栈方式分配内存空间。当程序执行到超出该变量的作用域时,就释放它所占用的内存空间,其值也随之消失了。声明一个自动存储类型的变量是在变量类型前面加上关键字auto。若自动存储类型的变量是在函数内或语句块中声明的,则可省略关键字auto。静态类型(static)一种局部变量。静态类型变量在内存中是以固定地址存放的,不是以堆栈方式存放的。只要程序在继续执行,它的值就有效,不会随它所在的函数或语句块的结束而消失。声明一个静态类型的变量是在变量类型前面加上关键字static。需要说明的是:(1)静态类型的变
40、量均有确定的初值,当声明变量时没有指定其初值,则编译器将其初值置为0。(2)程序中声明的全局变量是静态存储类型,若在全局变量前加一个static,使该变量只在这个源程序文件内使用,称之为全局静态变量或静态全局变量。若一个程序由一个文件组成,在声明全局变量时,有无static并没有区别,但若多个文件组成一个程序时,加与不加static,其作用完全不同。静态全局变量对组成该程序的其他源文件是无效的,它能很好地解决在程序多文件组织中全局变量的重名问题。(3)静态函数是在某个函数声明前加上static,它的目的是使该函数只在声明的源文件中使用,对于其他源文件则无效。,1.8作用域和存储类型,寄存器类型
41、(register)使用关键字register声明寄存器类型的变量的目的是将所声明的变量放入寄存器内,从而加快程序的运行速度。但有时在使用这种声明时,若系统寄存器已经被其他数据占据时,寄存器类型的变量就会被系统自动默认为auto变量。外部类型(extern)使用extern声明的变量称为外部变量,一般指定义在本程序外部的变量。某个变量被声明成外部变量时,不必再次为它分配内存就可以在本程序中引用这个变量。只有在两种情况下要使用外部变量:在同一个源文件中,若定义的变量使用在前,声明在后,这时在使用前要声明为外部变量。当由多个文件组成一个完整的程序时,在一个源程序文件中完全定义的变量要被其他若干个源
42、文件引用时,引用的文件中要使用extern声明外部变量。外部变量对不同源文件中或函数之间的数据传递特别有用。这种能被许多函数共享的外部变量,其数据值的任何一次改变,都将影响到所有引用此变量的函数的执行结果,其危险性是显而易见的。,1.9预处理,宏定义命令 用#define可以定义一个符号常量,如:#define PI 3.141593#define就是宏定义命令,它的作用是将3.141593用PI代替;PI称为宏名。需要注意的是:(1)#define、PI和3.141593之间一定要有空格,且一般将宏名定义成大写,以便与普通标识符相区别。(2)宏被定义后,使用下列命令后可重新定义:#undef
43、 宏名(3)一个定义过的宏名可以用来定义其它新的宏。(4)宏还可以带参数。文件包含命令 所谓“文件包含”是指将另一个源文件的内容合并到源程序中。C/C+语言提供了#include命令用来实现文件包含的操作,它有下列两种格式:#include#include“文件名”文件名一般是以.h为扩展名,因而称它为“头文件”,如前面的程序中iostream.h是头文件的文件名。文件包含的两种格式中,第一种格式是将文件名用尖括号“”括起来的,用来包含那些由系统提供的并放在指定子目录中的头文件。第二种格式是将文件名用双引号括起来的,用来包含那些由用户定义的放在当前目录或其他目录下的头文件或其他源文件。,1.9
44、预处理,条件编译命令 有时希望根据一定的条件去编译源文件的不同部分,就是“条件编译”。条件编译使得同一源程序在不同的编译条件下得到不同的目标代码。有几种常用的形式:(1)第一种形式#ifdef#else#endif是由若干条预处理命令或语句组成的。含义是:如果标识符已被#define命令定义过,则编译,否则编译。(2)第二种形式#ifndef#else#endif与前一种形式的区别仅在于,如果标识符没有被#define命令定义过,则编译,否则就编译。(3)第三种形式#if#elif.#else#endif含义是,如果为“真”就编译,否则如果为“真”就编译,.,如果各表达式都不为“真”就编译。,