数据类型、运算符与表达式.doc
3 数据类型、运算符与表达式 13.1 语言的数据类型 13.2 常量与变量 33.2.1 常量和符号常量 33.2.2 变量 33.3 整型数据 43.3.1 整型常量的表示方法 43.3.2 整型变量 53.4 实型数据 73.4.1 实型常量的表示方法 73.4.2 实型变量 83.4.3 实型常数的类型 93.5 字符型数据 93.5.1 字符常量 93.5.2 转义字符 93.5.3 字符变量 103.5.4 字符数据在内存中的存储形式及使用方法 103.5.5 字符串常量 113.5.6 符号常量 123.6 变量赋初值 123.7 各类数值型数据之间的混合运算 133.8 算术运算符和算术表达式 143.8.1 C运算符简介 143.8.2 算术运算符和算术表达式 153.9 赋值运算符和赋值表达式 173.10 逗号运算符和逗号表达式 183.11 小结 193.11.1 的数据类型 193.11.2 基本类型的分类及特点 193.11.3 常量后缀 193.11.4 常量类型 193.11.5 数据类型转换 193.11.6 运算符优先级和结合性 203.11.7 表达式 203 数据类型、运算符与表达式3.1 语言的数据类型在第一章中,我们已经看到程序中使用的各种变量都应预先加以定义,即先定义,后使用。对变量的定义可以包括三个方面:?数据类型 ?存储类型?作用域在本章中,我们只介绍数据类型的说明。其它说明在以后各章中陆续介绍。所谓数据类型是按被定义变量的性质,表示形式,占据存储空间的多少,构造特点来划分的。在语言中,数据类型可分为:基本数据类型,构造数据类型,指针类型,空类型四大类。1. 基本数据类型:基本数据类型最主要的特点是,其值不可以再分解为其它类型。也就是说,基本数据类型是自我说明的。2. 构造数据类型:构造数据类型是根据已定义的一个或多个数据类型用构造的方法来定义的。也就是说,一个构造类型的值可以分解成若干个“成员”或“元素”。每个“成员”都是一个基本数据类型或又是一个构造类型。在C语言中,构造类型有以下几种: ?数组类型 ?结构体类型 ?共用体(联合)类型3. 指针类型:指针是一种特殊的,同时又是具有重要作用的数据类型。其值用来表示某个变量在内存储器中的地址。虽然指针变量的取值类似于整型量,但这是两个类型完全不同的量,因此不能混为一谈。4. 空类型:在调用函数值时,通常应向调用者返回一个函数值。这个返回的函数值是具有一定的数据类型的,应在函数定义及函数说明中给以说明,例如在例题中给出的max函数定义中,函数头为:int max(int a,int b);其中“int ”类型说明符即表示该函数的返回值为整型量。又如在例题中,使用了库函数sin,由于系统规定其函数返回值为双精度浮点型,因此在赋值语句s=sin (x);中,s 也必须是双精度浮点型,以便与sin函数的返回值一致。所以在说明部分,把s说明为双精度浮点型。但是,也有一类函数,调用后并不需要向调用者返回函数值,这种函数可以定义为“空类型”。其类型说明符为void。在后面函数中还要详细介绍。在本章中,我们先介绍基本数据类型中的整型、浮点型和字符型。其余类型在以后各章中陆续介绍。3.2 常量与变量对于基本数据类型量,按其取值是否可改变又分为常量和变量两种。在程序执行过程中,其值不发生改变的量称为常量,其值可变的量称为变量。它们可与数据类型结合起来分类。例如,可分为整型常量、整型变量、浮点常量、浮点变量、字符常量、字符变量、枚举常量、枚举变量。在程序中,常量是可以不经说明而直接引用的,而变量则必须先定义后使用。整型量包括整型常量、整型变量。3.2.1 常量和符号常量在程序执行过程中,其值不发生改变的量称为常量。? 直接常量(字面常量):? 整型常量:12、0、-3;? 实型常量:4.6、-1.23;? 字符常量:a、b。? 标识符:用来标识变量名、符号常量名、函数名、数组名、类型名、文件名的有效字符序列。? 符号常量:用标示符代表一个常量。在语言中,可以用一个标识符来表示一个常量,称之为符号常量。符号常量在使用之前必须先定义,其一般形式为:#define 标识符 常量其中#define也是一条预处理命令(预处理命令都以"#"开头),称为宏定义命令(在后面预处理程序中将进一步介绍),其功能是把该标识符定义为其后的常量值。一经定义,以后在程序中所有出现该标识符的地方均代之以该常量值。? 习惯上符号常量的标识符用大写字母,变量标识符用小写字母,以示区别。【例3.1】符号常量的使用。#define PRICE 30main() int num,total;num=10;total=num* PRICE;printf(“total=%d”,total);? 用标识符代表一个常量,称为符号常量。? 符号常量与变量不同,它的值在其作用域内不能改变,也不能再被赋值。? 使用符号常量的好处是:? 含义清楚;? 能做到“一改全改”。3.2.2 变量其值可以改变的量称为变量。一个变量应该有一个名字,在内存中占据一定的存储单元。变量定义必须放在变量使用之前。一般放在函数体的开头部分。要区分变量名和变量值是两个不同的概念。3.3 整型数据3.3.1 整型常量的表示方法整型常量就是整常数。在语言中,使用的整常数有八进制、十六进制和十进制三种。1) 十进制整常数:十进制整常数没有前缀。其数码为09。以下各数是合法的十进制整常数:237、-568、65535、1627;以下各数不是合法的十进制整常数:023 (不能有前导0)、23D (含有非十进制数码)。在程序中是根据前缀来区分各种进制数的。因此在书写常数时不要把前缀弄错造成结果不正确。2) 八进制整常数:八进制整常数必须以0开头,即以0作为八进制数的前缀。数码取值为07。八进制数通常是无符号数。以下各数是合法的八进制数:015(十进制为13)、0101(十进制为65)、(十进制为65535);以下各数不是合法的八进制数:256(无前缀0)、03A2(包含了非八进制数码)、-0127(出现了负号)。3) 十六进制整常数:十六进制整常数的前缀为0X或0x。其数码取值为09,AF或af。以下各数是合法的十六进制整常数:0X2A(十进制为42)、0XA0 (十进制为160)、0XFFFF (十进制为65535);以下各数不是合法的十六进制整常数:5A (无前缀0X)、0X3H (含有非十六进制数码)。4) 整型常数的后缀:在16位字长的机器上,基本整型的长度也为16位,因此表示的数的范围也是有限定的。十进制无符号整常数的范围为065535,有符号数为-32768+32767。八进制无符号数的表示范围为0。十六进制无符号数的表示范围为0X00XFFFF或0x00xFFFF。如果使用的数超过了上述范围,就必须用长整型数来表示。长整型数是用后缀“L”或“l”来表示的。例如:十进制长整常数:158L (十进制为158)、L (十进制为);八进制长整常数:012L (十进制为10)、077L (十进制为63)、L (十进制为65536);十六进制长整常数:0X15L (十进制为21)、0XA5L (十进制为165)、0X10000L (十进制为65536)。长整数158L和基本整常数158 在数值上并无区别。但对158L,因为是长整型量,编译系统将为它分配4个字节存储空间。而对158,因为是基本整型,只分配2 个字节的存储空间。因此在运算和输出格式上要予以注意,避免出错。无符号数也可用后缀表示,整型常数的无符号数的后缀为“U”或“u”。例如:358u,0x38Au,235Lu均为无符号数。前缀,后缀可同时使用以表示各种类型的数。如0XA5Lu表示十六进制无符号长整数A5,其十进制为165。3.3.2 整型变量1. 整型数据在内存中的存放形式如果定义了一个整型变量i:int i;i=10;i 100 0 0 0 0 0 0 0 0 0 0 0 1 0 1 0数值是以补码表示的:? 正数的补码和原码相同;? 负数的补码:将该数的绝对值的二进制形式按位取反再加1。例如:求-10的补码:10的原码:0 0 0 0 0 0 0 0 0 0 0 0 1 0 1 0 取反:1 1 1 1 1 1 1 1 1 1 1 1 0 1 0 1再加1,得-10的补码:1 1 1 1 1 1 1 1 1 1 1 1 0 1 1 0由此可知,左面的第一位是表示符号的。2. 整型变量的分类1) 基本型:类型说明符为int,在内存中占2个字节。2) 短整量:类型说明符为short int或short。所占字节和取值范围均与基本型相同。3) 长整型:类型说明符为long int或long,在内存中占4个字节。4) 无符号型:类型说明符为unsigned。无符号型又可与上述三种类型匹配而构成:? 无符号基本型:类型说明符为unsigned int或unsigned。? 无符号短整型:类型说明符为unsigned short。? 无符号长整型:类型说明符为unsigned long。各种无符号类型量所占的内存空间字节数与相应的有符号类型量相同。但由于省去了符号位,故不能表示负数。有符号整型变量:最大表示327670 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1无符号整型变量:最大表示655351 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 下表列出了Turbo C中各类整型量所分配的内存字节数及数的表示范围。类型说明符 数的范围 字节数int -3276832767 即-215(215-1) 2unsigned int 065535 即0(216-1) 2short int -3276832767 即-215(215-1) 2unsigned short int 065535 即0(216-1) 2long int -即-231(231-1) 4unsigned long 0 即0(232-1) 4以13为例:int型:00 00 00 00 00 00 11 01short int型:00 00 00 00 00 00 11 01long int型:00 00 00 00 00 00 00 00 00 00 00 00 00 00 11 01unsigned int型:00 00 00 00 00 00 11 01unsigned short int型:00 00 00 00 00 00 11 01unsigned long int型:00 00 00 00 00 00 00 00 00 00 00 00 00 00 11 013. 整型变量的定义变量定义的一般形式为:类型说明符 变量名标识符,变量名标识符,.;例如: int a,b,c; (a,b,c为整型变量) long x,y; (x,y为长整型变量)unsigned p,q; (p,q为无符号整型变量)在书写变量定义时,应注意以下几点:? 允许在一个类型说明符后,定义多个相同类型的变量。各变量名之间用逗号间隔。类型说明符与变量名之间至少用一个空格间隔。? 最后一个变量名之后必须以“;”号结尾。? 变量定义必须放在变量使用之前。一般放在函数体的开头部分。【例3.2】整型变量的定义与使用。main()int a,b,c,d;unsigned u;a=12;b=-24;u=10;c=a+u;d=b+u;printf(“a+u=%d,b+u=%dn”,c,d);4. 整型数据的溢出【例3.3】整型数据的溢出。main() int a,b; a=32767; b=a+1; printf("%d,%dn",a,b);32767:0 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1-327681 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0【例3.4】main() long x,y; int a,b,c,d; x=5; y=6; a=7; b=8; c=x+a; d=y+b; printf("c=x+a=%d,d=y+b=%dn",c,d);从程序中可以看到:x, y是长整型变量,a, b是基本整型变量。它们之间允许进行运算,运算结果为长整型。但c,d被定义为基本整型,因此最后结果为基本整型。本例说明,不同类型的量可以参与运算并相互赋值。其中的类型转换是由编译系统自动完成的。有关类型转换的规则将在以后介绍。3.4 实型数据3.4.1 实型常量的表示方法实型也称为浮点型。实型常量也称为实数或者浮点数。在语言中,实数只采用十进制。它有二种形式:十进制小数形式,指数形式。1) 十进制数形式:由数码0 9和小数点组成。例如:0.0、25.0、5.789、0.13、5.0、300.、-267.8230等均为合法的实数。注意,必须有小数点。2) 指数形式:由十进制数,加阶码标志“e”或“E”以及阶码(只能为整数,可以带符号)组成。其一般形式为:a E n(a为十进制数,n为十进制整数)其值为 a*10n。如:2.1E5 (等于2.1*105)3.7E-2 (等于3.7*10-2)0.5E7 (等于0.5*107)-2.8E-2 (等于-2.8*10-2)以下不是合法的实数:345 (无小数点)E7 (阶码标志E之前无数字)-5 (无阶码标志)53.-E3 (负号位置不对)2.7E (无阶码)标准允许浮点数使用后缀。后缀为“f”或“F”即表示该数为浮点数。如356f和356.是等价的。【例3.5】说明了这种情况。main() printf("%fn ",356.); printf("%fn ",356); printf("%fn ",356f);3.4.2 实型变量1. 实型数据在内存中的存放形式实型数据一般占4个字节(32位)内存空间。按指数形式存储。实数3.14159在内存中的存放形式如下:+ . 1 数符 小数部分 指数? 小数部分占的位(bit)数愈多,数的有效数字愈多,精度愈高。? 指数部分占的位数愈多,则能表示的数值范围愈大。2. 实型变量的分类实型变量分为:单精度(float型)、双精度(double型)和长双精度(long double型)三类。在Turbo C中单精度型占4个字节(32位)内存空间,其数值范围为3.4E-383.4E+38,只能提供七位有效数字。双精度型占8 个字节(64位)内存空间,其数值范围为1.7E-3081.7E+308,可提供16位有效数字。类型说明符 比特数(字节数) 有效数字 数的范围float 32(4) 67 10-371038 double 64(8) 1516 10-30710308 long double 128(16) 1819 10-4931 实型变量定义的格式和书写规则与整型相同。例如: float x,y; (x,y为单精度实型量)double a,b,c; (a,b,c为双精度实型量)3. 实型数据的舍入误差由于实型变量是由有限的存储单元组成的,因此能提供的有效数字总是有限的。如下例。【例3.6】实型数据的舍入误差。main()float a,b;a=.789e5;b=a+20printf("%fn",a);printf("%fn",b);注意:1.0/3*3的结果并不等于1。【例3.7】main() float a; double b; a=33333.33333; b=33333.333; printf("%fn%fn",a,b);? 从本例可以看出,由于a 是单精度浮点型,有效位数只有七位。而整数已占五位,故小数二位后之后均为无效数字。? b 是双精度型,有效位为十六位。但Turbo C 规定小数后最多保留六位,其余部分四舍五入。3.4.3 实型常数的类型实型常数不分单、双精度,都按双精度double型处理。3.5 字符型数据字符型数据包括字符常量和字符变量。3.5.1 字符常量字符常量是用单引号括起来的一个字符。例如:'a'、'b'、'='、'+'、'?'都是合法字符常量。在语言中,字符常量有以下特点:1) 字符常量只能用单引号括起来,不能用双引号或其它括号。2) 字符常量只能是单个字符,不能是字符串。3) 字符可以是字符集中任意字符。但数字被定义为字符型之后就不能参与数值运算。如'5'和5 是不同的。'5'是字符常量,不能参与运算。3.5.2 转义字符 转义字符是一种特殊的字符常量。转义字符以反斜线""开头,后跟一个或几个字符。转义字符具有特定的含义,不同于字符原有的意义,故称“转义”字符。例如,在前面各例题printf函数的格式串中用到的“n”就是一个转义字符,其意义是“回车换行”。转义字符主要用来表示那些用一般字符不便于表示的控制代码。常用的转义字符及其含义转义字符 转义字符的意义 ASCII代码 n 回车换行 10 t 横向跳到下一制表位置 9 b 退格 8 r 回车 13 f 走纸换页 12 反斜线符"" 92 ' 单引号符 39” 双引号符 34 a 鸣铃 7 ddd 13位八进制数所代表的字符 xhh 12位十六进制数所代表的字符 广义地讲,语言字符集中的任何一个字符均可用转义字符来表示。表中的ddd和xhh正是为此而提出的。ddd和hh分别为八进制和十六进制的ASCII代码。如101表示字母"A" ,102表示字母"B",134表示反斜线,XOA表示换行等。【例3.8】转义字符的使用。main() int a,b,c; a=5; b=6; c=7; printf(“ ab ctderfn”);printf(“hijktLbMn”);3.5.3 字符变量字符变量用来存储字符常量,即单个字符。字符变量的类型说明符是char。字符变量类型定义的格式和书写规则都与整型变量相同。例如:char a,b;3.5.4 字符数据在内存中的存储形式及使用方法每个字符变量被分配一个字节的内存空间,因此只能存放一个字符。字符值是以ASCII码的形式存放在变量的内存单元之中的。如x的十进制ASCII码是120,y的十进制ASCII码是121。对字符变量a,b赋予'x'和'y'值:a='x' b='y'实际上是在a,b两个单元内存放120和121的二进制代码:a:0 1 1 1 1 0 0 0b:0 1 1 1 1 0 0 1所以也可以把它们看成是整型量。语言允许对整型变量赋以字符值,也允许对字符变量赋以整型值。在输出时,允许把字符变量按整型量输出,也允许把整型量按字符量输出。 整型量为二字节量,字符量为单字节量,当整型量按字符型量处理时,只有低八位字节参与处理。【例3.9】向字符变量赋以整数。main() char a,b; a=120; b=121; printf("%c,%cn",a,b);printf("%d,%dn",a,b); 本程序中定义a,b为字符型,但在赋值语句中赋以整型值。从结果看,a,b值的输出形式取决于printf函数格式串中的格式符,当格式符为"c"时,对应输出的变量值为字符,当格式符为"d"时,对应输出的变量值为整数。【例3.10】main() char a,b; a='a' b='b' a=a-32; b=b-32; printf("%c,%cn%d,%dn",a,b,a,b);本例中,a,b被说明为字符变量并赋予字符值,语言允许字符变量参与数值运算,即用字符的ASCII 码参与运算。由于大小写字母的ASCII 码相差32,因此运算后把小写字母换成大写字母。然后分别以整型和字符型输出。3.5.5 字符串常量字符串常量是由一对双引号括起的字符序列。例如: "CHINA" , “C program” , "$12.5" 等都是合法的字符串常量。字符串常量和字符常量是不同的量。它们之间主要有以下区别:1) 字符常量由单引号括起来,字符串常量由双引号括起来。2) 字符常量只能是单个字符,字符串常量则可以含一个或多个字符。3) 可以把一个字符常量赋予一个字符变量,但不能把一个字符串常量赋予一个字符变量。在语言中没有相应的字符串变量。这是与BASIC 语言不同的。但是可以用一个字符数组来存放一个字符串常量。在数组一章内予以介绍。4) 字符常量占一个字节的内存空间。字符串常量占的内存字节数等于字符串中字节数加1。增加的一个字节中存放字符"0" (ASCII码为0)。这是字符串结束的标志。例如:字符串 "C program" 在内存中所占的字节为:C p r o g r a m 0字符常量'a'和字符串常量"a"虽然都只有一个字符,但在内存中的情况是不同的。'a'在内存中占一个字节,可表示为:a"a"在内存中占二个字节,可表示为:a 0 3.6 变量赋初值在程序中常常需要对变量赋初值,以便使用变量。语言程序中可有多种方法为变量提供初值。本小节先介绍在作变量定义的同时给变量赋以初值的方法。这种方法称为初始化。在变量定义中赋初值的一般形式为: 类型说明符 变量1= 值1,变量2= 值2,;例如:int a=3;int b,c=5; floatx=3.2,y=3f,z=0.75;charch1='K',ch2='P'应注意,在定义中不允许连续赋值,如a=b=c=5是不合法的。【例3.11】main() inta=3,b,c=5; b=a+c;< BR> printf("a=%d,b=%d,c=%dn",a,b,c); 3.7 各类数值型数据之间的混合运算变量的数据类型是可以转换的。转换的方法有两种,一种是自动转换,一种是强制转换。自动转换发生在不同数据类型的量混合运算时,由编译系统自动完成。自动转换遵循以下规则:1) 若参与运算量的类型不同,则先转换成同一类型,然后进行运算。2) 转换按数据长度增加的方向进行,以保证精度不降低。如int型和long型运算时,先把int量转成long型后再进行运算。3) 所有的浮点运算都是以双精度进行的,即使仅含float单精度量运算的表达式,也要先转换成double型,再作运算。4) char型和short型参与运算时,必须先转换成int型。5) 在赋值运算中,赋值号两边量的数据类型不同时,赋值号右边量的类型将转换为左边量的类型。如果右边量的数据类型长度左边长时,将丢失一部分数据,这样会降低精度,丢失的部分按四舍五入向前舍入。下图表示了类型自动转换的规则。【例3.12】main() float PI=3.14159; int s,r=5; s=r*r*PI; printf("s=%dn",s);本例程序中,PI为实型;s,r为整型。在执行s=r*r*PI语句时,r和PI都转换成double型计算,结果也为double型。但由于s为整型,故赋值结果仍为整型,舍去了小数部分。强制类型转换强制类型转换是通过类型转换运算来实现的。其一般形式为: (类型说明符) (表达式)其功能是把表达式的运算结果强制转换成类型说明符所表示的类型。例如: (float) a 把a转换为实型 (int)(x+y) 把x+y的结果转换为整型在使用强制转换时应注意以下问题:1) 类型说明符和表达式都必须加括号(单个变量可以不加括号),如把(int)(x+y)写成(int)x+y则成了把x转换成int型之后再与y相加了。2) 无论是强制转换或是自动转换,都只是为了本次运算的需要而对变量的数据长度进行的临时性转换,而不改变数据说明时对该变量定义的类型。【例3.13】main() float f=5.75; printf("(int)f=%d,f=%fn",(int)f,f); 本例表明,f虽强制转为int型,但只在运算中起作用,是临时的,而f本身的类型并不改变。因此,(int)f的值为 5(删去了小数)而f的值仍为5.75。3.8 算术运算符和算术表达式 语言中运算符和表达式数量之多,在高级语言中是少见的。正是丰富的运算符和表达式使语言功能十分完善。这也是语言的主要特点之一。 语言的运算符不仅具有不同的优先级,而且还有一个特点,就是它的结合性。在表达式中,各运算量参与运算的先后顺序不仅要遵守运算符优先级别的规定,还要受运算符结合性的制约,以便确定是自左向右进行运算还是自右向左进行运算。这种结合性是其它高级语言的运算符所没有的,因此也增加了语言的复杂性。3.8.1 C运算符简介语言的运算符可分为以下几类:1. 算术运算符:用于各类数值运算。包括加(+)、减(-)、乘(*)、除(/)、求余(或称模运算,%)、自增(+)、自减(-)共七种。2. 关系运算符:用于比较运算。包括大于(>)、小于(<)、等于(= =)、大于等于(>=)、小于等于(<=)和不等于(!=)六种。3. 逻辑运算符:用于逻辑运算。包括与(&&)、或(|)、非(!)三种。4. 位操作运算符:参与运算的量,按二进制位进行运算。包括位与(&)、位或(|)、位非()、位异或()、左移(<<)、右移(>>)六种。5. 赋值运算符:用于赋值运算,分为简单赋值(=)、复合算术赋值(+=,-=,*=,/=,%=)和复合位运算赋值(&=,|=,=,>>=,<<=)三类共十一种。6. 条件运算符:这是一个三目运算符,用于条件求值(?:)。7. 逗号运算符:用于把若干表达式组合成一个表达式(,)。8. 指针运算符:用于取内容(*)和取地址(&)二种运算。9. 求字节数运算符:用于计算数据类型所占的字节数(sizeof)。10. 特殊运算符:有括号(),下标,成员(,.)等几种。3.8.2 算术运算符和算术表达式1. 基本的算术运算符? 加法运算符“+”:加法运算符为双目运算符,即应有两个量参与加法运算。如a+b,4+8等。具有右结合性。? 减法运算符“-”:减法运算符为双目运算符。但“-”也可作负值运算符,此时为单目运算,如-x,-5