第三章
本章要点
数据的描述规则
数据的操作规则
主要内容
3.1 C的数据类型
3.2 常量与变量
3.3 整型数据
3.4 浮点型数据运行
3.5 字符型数据
主要内容
3.6变量赋初值
3.7 各类数值型数据间的混合运算
3.8 算术运算符和算术表达式
3.9 赋值运算符和赋值表达式
3.10 逗号运算符和逗号表达式
C程序设计(第三版) http://ccf.tsinghua.edu.cn 5
3.1 C的数据类型
C语言提供了以下一些数据类型。
数据类型 构造类型指针类型空类型( 无值类型) void
枚举类型 enum
数组类型结构类型 struct
联合类型 union
基本类型整型 int
字符型 char
实型(浮点型)
单精度实型
float
双精度实型
double
C程序设计(第三版) http://ccf.tsinghua.edu.cn 6
3.2 常量与变量
3.2.1 常量和符号常量
在程序运行过程中,其值不能被改变的量称为常量
常量区分为不同的类型:
整型 100,125,-100,0
实型 3.14,0.125,-3.789
字符型 ‘ a?,?b?,‘ 2?
字符串 ‘ a?,?ab?,‘ 1232?
C程序设计(第三版) http://ccf.tsinghua.edu.cn 7
例 3.1 符号常量的使用
#define PRICE 30
#include <stdio.h>
void main ( )
{
int num,total;
num=10;
total=num * PRICE;
printf(″total=%d \n″,total);
} 符号常量,用一个标识符代表一个常量。
符号常量的值在其作用域内不能改变,
也不能再被赋值。
运行结果,total=300
说明,程序中用 #define命令行定义 PRICE
代表常量 30,此后凡在本文件中出现的
PRICE都代表 30,可以和常量一样进行运算说明,如再用赋值语句给 PRICE赋值是错的
PRICE=40; /* 错误,不能给符号常量赋值
C程序设计(第三版) http://ccf.tsinghua.edu.cn 8
3.2 常量与变量
3.2.2 变量
变量代表内存中具有特定属性的一个存储单元,它用来存放数据,这就是变量的值,在程序运行期间,这些值是可以改变的。
变量名实际上是一个以一个名字对应代表一个地址,在对程序编译连接时由编译系统给每一个变量名分配对应的内存地址。从变量中取值,实际上是通过变量名找到相应的内存地址,从该存储单元中读取数据。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 9
3.2 常量与变量变量命名的规定,C语言规定标识符只能由字母、数字和下划线三种字符组成,且第一个字符必须为字母或下划线。
例,sum,_total,month,Student_name,
lotus_1_2_3,BASIC,li_ling
M.D.John,¥ 123,3D64,a>b
C程序设计(第三版) http://ccf.tsinghua.edu.cn 10
3.2 常量与变量注意:
编译系统将大写字母和小写字母认为是两个不同的字符。
建议变量名的长度最好不要超过 8个字符。
在选择变量名和其它标识符时,应注意做到? 见名知意?,即选有含意的英文单词
(或其缩写)作标识符。
要求对所有用到的变量作强制定义,也就是? 先定义,后使用? 。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 11
3.3 整型数据
3.3.1整型常量的表示方法整型常量即整常数。在C语言中,整常数可用以下三种形式表示:
(1)十进制整数。
如,123,-456.4。
(2)八进制整数。以 0头的数是八进制数。
如,0123表示八进制数 123,等于十进制数
83,-011表示八进制数 -11,即十进制数 -9。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 12
3.3 整型数据
(3)十六进制整数。以 0x开头的数是 16进制数。
如,0x123,代表 16进制数 123,等于十进制数
291。 -0x12等于十进制数- 10。
3.3.2 整型变量
(1)整型数据在内存中的存放形式数据在内存中是以二进制形式存放的。
如,int i; /* 定义为整型变量 */
i=10; /* 给 i赋以整数 10 */
C程序设计(第三版) http://ccf.tsinghua.edu.cn 13
3.3 整型数据注意:
十进制数 10的二进制形式为 1010,Turbo
C 2.0和 Turbo C++ 3.0为一个整型变量在内存中分配 2个字节的存储单元 (不同的编译系统为整型数据分配的字节数是不相同的,VC++ 6.0则分配 4个字节)。
数值是以补码 (complement) 表示的。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 14
3.3 整型数据
(2)整型变量的分类共六种有符号基本整型有符号短整型有符号长整型无符号基本整型无符号短整型无符号长整型
(signed)int
(signed)short (int )
(signed) long (int)
unsigned int
unsigned short (int)
unsigned long (int)
注意,括号表示其中的内容是可选的,
C程序设计(第三版) http://ccf.tsinghua.edu.cn 15
3.3 整型数据
整数类型的有关数据:
类型 类型说明符 长度 数的范围基本型 int 2字节 -32768~ 32767
短整型 short 2字节 -215~ 215-1
长整型 long 4字节 -231~ 231-1
无符号整型 unsigned 2字节 0~ 65535
无符号短整型 unsigned short 2字节 0~ 65535
无符号长整型 unsigned long 4字节 0~( 232-1)
C程序设计(第三版) http://ccf.tsinghua.edu.cn 16
3.3 整型数据例如:整数 13在内存中实际存放的情况:
C程序设计(第三版) http://ccf.tsinghua.edu.cn 17
3.3 整型数据
(3)整型变量的定义,
C规定在程序中所有用到的变量都必须在程序中定义,即? 强制类型定义? 。
例如,
int a,b(指定变量a、b为整型)
unsigned short c,d;(指定变量c、d为无符号短整型)
long e,f;(指定变量e、f为长整型)
C程序设计(第三版) http://ccf.tsinghua.edu.cn 18
例 3.2 整型变量的定义与使用
#include <stdio.h>
void main()
{ int a,b,c,d; /* 指定a、b、c、d为整
*/
unsigned u; / *
*/
a= 12;b= -24;u= 10;
c=a+u;d=b+u;
printf( ″a+u=%d,b+u=%d\n ″
,c,d);
}
说明,可以看到不同种类的整型数据可以进行算术运算运行结果,a+u=22
,b+u=-14
C程序设计(第三版) http://ccf.tsinghua.edu.cn 19
例 3.3 整型数据的溢出
#include <stdio.h>
void main()
{int a,b;
a=32767;
b=a+1;
printf(“%d,%d \n”,a,b);
} 说明,数值是以补码表示的。一个整型变量只能容纳 -32768~ 32767范围内的数,无法表示大于 32767或小于 -32768的数。遇此情况就发生“溢出”。
运行结果,32767,-32768
C程序设计(第三版) http://ccf.tsinghua.edu.cn 20
3.3 整型数据
3.3.3 整型常量的类型
(1)一个整数,如果其值在 -32768~ +32767范围内,认为它是 int型,它可以赋值给 int型和 long int型变量。
(2) 一个整数,如果其值超过了上述范围,
而在 -2147483637~ +2147483647范围内,则认为它是为长整型。可以将它赋值给一个
long int型变量。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 21
3.3 整型数据
(3) 如果所用的 C版本(如 Turbo C)分配给
short int与 int型数据在内存中占据的长度相同,则它的表数范围与 int型相同。因此一个 int型的常量同时也是一个 short int型常量,可以赋给 int型或 short int型变量。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 22
3.3 整型数据
(4) 一个整常量后面加一个字母 u或 U,认为是 unsigned int型,如 12345u,在内存中按 unsigned int规定的方式存放 (存储单元中最高位不作为符号位,而用来存储数据 )。如果写成 -12345u,则先将 -12345
转换成其补码 53191,然后按无符号数存储。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 23
3.3 整型数据
(5) 在一个整常量后面加一个字母 l或 L
,则认为是 long int型常量。
例如,123l.432L.0L
用于函数调用中。
如果函数的形参为 long int型,则要求实参也为 long int型。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 24
3.4 浮点型数据
3.4.1浮点型常量的表示方法两种表示形式小数指数
0.123
3e-3
注意,字母 e(或 E)之前必须有数字,且 e后面的指数必须为整数
1e3,1.8e-3,-123e-6,-.1e-3
e3,2.1e3.5,.e3,e
C程序设计(第三版) http://ccf.tsinghua.edu.cn 25
3.4 浮点型数据规范化的指数形式:
在字母 e(或 E)之前的小数部分中,小数点左边应有一位(且只能有一位)非零的数字,
例如,123.456可以表示为:
123.456e0,12.3456e1,1.23456e2,
0.123456e3,0.0123456e4,0.00123456e
其中的 1.23456e3称为? 规范化的指数形式
。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 26
3.4 浮点型数据
3.4.2 浮点型变量
(1)浮点型数据在内存中的存放形式一个浮点型数据一般在内存中占 4
个字节 (32位 )。与整型数据的存储方式不同,浮点型数据是按照指数形式存储的。系统把一个浮点型数据分成小数部分和指数部分,分别存放。指数部分采用规范化的指数形式。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 27
类型 位数 数的范围 有效数字
float 32 10-37 ~ 1038 6~7 位
double型 64 10-307~10308 15~16位
long double 128 10-4931~104932 18~19位
3.4 浮点型数据
(2) 浮点型变量的分类浮点型变量分为单精度( float型)、双精度
( double型)和长双精度型( long double)三类形式。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 28
例 3.4 浮点型数据的舍入误差
#include <stdio.h>
void main()
{float a,b;
a = 123456.789e5;
b = a + 20 ;
printf(“%f \n”,b);
}说明,一个浮点型变量只能保证的有效数字是 7位有效数字,后面的数字是无意义的,并不准确地表示该数。应当避免将一个很大的数和一个很小的数直接相加或相减,否则就会“丢失”小的数运行结果,123456.789e5
C程序设计(第三版) http://ccf.tsinghua.edu.cn 29
3.4 浮点型数据
3.4.3 浮点型常量的类型
C编译系统将浮点型常量作为双精度来处理。
例如,f = 2.45678 * 4523.65
系统先把 2.45678和 4523.65作为双精度数
,然后进行相乘的运算,得到的乘也是一个双精度数。最后取其前 7位赋给浮点型变量 f
。如是在数的后面加字母 f或 F(如 1.65f,
654.87F),这样编译系统就会把它们按单精度( 32位)处理。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 30
3.5 字符型数据
3.5.1 字符常量
(1)用单引号包含的一个字符是字符型常量
(2)只能包含一个字符
‘a’,’A’,‘1’
‘abc’,,a”?
C程序设计(第三版) http://ccf.tsinghua.edu.cn 31
3.5 字符型数据有些以? \”开头的特殊字符称为转义字符
\n 换行
\t 横 向跳格
\r 回 车
\\ 反 斜杠
\ddd ddd表示 1到 3位八进制数字
\xhh hh表示 1到 2位十六进制数字
C程序设计(第三版) http://ccf.tsinghua.edu.cn 32
例 3.5 转义字符的使用
#include <stdio.h>
void main()
{printf( ″ ab c \t
de\rf\tg\n″ ) ;
printf(″h \ti\b\bj k\n″);
}
打印机上的显示结果:
fab c gde
h jik
显示屏上 的运行结果:
f gde
h k
C程序设计(第三版) http://ccf.tsinghua.edu.cn 33
3.5 字符型数据
3.5.2字符变量
字符型变量用来存放字符常量,注意只能放一个字符 。
字符变量的定义形式如下,char c1,c2;
在本函数中可以用下面语句对 c1,c2赋值:
c1= ‘ a?; c2= ‘ b? ;
一个字符变量在内存中占一个字节。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 34
3.5 字符型数据
3.5.3 字符数据在内存中的存储形式及其使用方法
一个字符常量存放到一个字符变量中,实际上并不是把该字符的字型放到内存中去,而是将该字符的相应的 ASCII代码放到存储单元中。这样使字符型数据和整型数据之间可以通用。
注意,
一个字符数据既可以以字符形式输出,也可以以整数形式输出。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 35
例 3.6 向字符变量赋以整数。
#include <stdio.h>
void main()
{char c1,c2;
c1=97;
c2=98;
printf(“%c %c \n”,c1,c2);
printf(“%d %d \n”,c1,c2);
}? 说明,在第3和第 4行中,将整数 97和 98分别赋给 c1和 c2,它的作用相当于以下两个赋值语句:
c1= ′ a ′;c2 = ′ b ′;
因为’ a’ 和’ b’ 的 ASCII码为 97和 98
运行结果:
a b
97 98
C程序设计(第三版) http://ccf.tsinghua.edu.cn 36
例 3.7 大小写字母的转换
#include <stdio.h>
void main()
{char c1,c2;
c1=’a’;
c2=’b’;
c1=c1-32;
c2=c2-32;
printf(,% c % c″,c1,c2);
}
说明,程序的作用是将两个小写字母 a和 b转换成大写字母 A和 B。从ASCII代码表中可以看到每一个小写字母比它相应的大写字母的 ASCII码大 32。C语言允许字符数据与整数直接进行算术运算。
运行结果,A B
C程序设计(第三版) http://ccf.tsinghua.edu.cn 37
3.5 字符型数据说明:
有些系统(如 Turbo C)将字符变量定义为 signed
char型。其存储单元中的最高位作为符号位,它的取值范围是 -128~ 127。如果在字符变量中存放一个
ASCII码为 0~ 127间的字符,由于字节中最高位为 0,
因此用 %d输出字符变量时,输出的是一个正整数。如果在字符变量中存放一个 ASCII码为 128~ 255间的字符,由于在字节中最高位为 1,用 %d格式符输出时,
就会得到一个负整数。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 38
3.5 字符型数据
3.5.4字符串常量
字符串常量是一对双撇号括起来的字符序列,
合法的字符串常量,
How do you do.”,“CHINA”,? a”,
$123.45”
可以输出一个字符串,如
printf(“How do you do.”);
C程序设计(第三版) http://ccf.tsinghua.edu.cn 39
3.5 字符型数据
a ’ 是字符常量,? a? 是字符串常量,二者不同。
如,假设C被指定为字符变量,char cc=’a’;
c=” a” ; c=” CHINA”;
结论,不能把一个字符串常量赋给一个字符变量。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 40
C规定:在每一个字符串常量的结尾加一个? 字符串结束标志?,以便系统据此判断字符串是否结束。
C规定以字符 ’ \0 ’ 作为字符串结束标志。如,如果有一个字符串常量”CHINA”,实际上在内存中是,C H I N A \0
它占内存单元不是5个字符,而是6个字符,
最后一个字符为’\0’。但在输出时不输出’
\0’。
3.5 字符型数据
C程序设计(第三版) http://ccf.tsinghua.edu.cn 41
3.6 变量赋初值字符串常量
(1)C语言允许在定义变量的同时使变量初始化如,int a=3; // 指定a为整型变量,初值为3
float f=3.56; // 指定f为浮点型变量,初值为3,56
char c= ‘a’; // 指定c为字符变量,初值为‘ a’
C程序设计(第三版) http://ccf.tsinghua.edu.cn 42
3.6 变量赋初值
(2)可以使被定义的变量的一部分赋初值。
如,int a,b,c=5; 表示指定a、b、c为整型变量,但只对c初始化,c的初值为5
(3)如果对几个变量赋以同一个初值,
应写成,int a=3,b=3,c=3; 表示a、b、c
的初值都是3。
不能写成 ∶ int a=b=c3;
注意,初始化不是在编译阶段完成的而是在程序运行时执行本函数时赋初值的,相当于有一个赋值语句。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 43
3.7 各类数值型数据间的混合运算混合运算,整型(包括 int,short,long)、浮点型(包括 float,double)可以混合运算。
在进行运算时,不同类型的数据要先转换成同一类型,然后进行运算,
说明:
这种类型转换是由系统自动进行的。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 44
3.8 术运算符和算术表达式
3.8.1 C运算符简介
C的运算符有以下几类:
(1)算术运算符 ( + - * / %)
(2)关系运算符 (><==>=<=!=)
(3)逻辑运算符 (!&& ||)
(4)位运算符 ( << >> ~ |∧ &)
(5)赋值运算符 (=及其扩展赋值运算符)
(6)条件运算符 (?:)
(7)逗号运算符 (,)
C程序设计(第三版) http://ccf.tsinghua.edu.cn 45
3.8 算术运算符和算术表达式
(8)指针运算符 ( *和&)
(9)求字节数运算符(sizeof)
(10)强制类型转换运算符( (类型) )
(11)分量运算符(,->)
(12)下标运算符([ ])
(13)其他 (如函数调用运算符())
C程序设计(第三版) http://ccf.tsinghua.edu.cn 46
3.8 算术运算符和算术表达式
3.8.2 算术运算符和算术表达式
(1)基本的算术运算符,
+ ( 加法运算符,或正值运算符。如,3+5、+3)
- ( 减法运算符,或负值运算符。如,5-2、-3)
* (乘法运算符。如,3 *5)
/ ( 除法运算符。如,5/3)
% ( 模运算符,或称求余运算符,%两侧均应为整型数据,如,7%4的值为3)。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 47
3.8 算术运算符和算术表达式
(2) 算术表达式和运算符的优先级与结合性基本的算术运算符,
用算术运算符和括号将运算对象 (也称操作数)
连接起来的、符合C语法规则的式子,称为C算术表达式。运算对象包括常量、变量、函数等。
例如,a *b/c-1,5+ ′a′
是一个合法的表达式
C程序设计(第三版) http://ccf.tsinghua.edu.cn 48
C语言规定了运算符的优先级和结合性。
在表达式求值时,先按运算符的优先级别高低次序执行,例如先乘除后加减。
C规定了各种运算符的结合方向 (结合性 )
算术运算符的结合方向为“自左至右”,即先左后右 。
3.8 算术运算符和算术表达式
C程序设计(第三版) http://ccf.tsinghua.edu.cn 49
3.8 算术运算符和算术表达式
(3)强制类型转换运算符可以利用强制类型转换运算符将一个表达式转换成所需类型。
一般形式,(类型名)(表达式)
例如:
(double)a 将a转换成 double类型
(int)(x+y) 将 x+y的值转换成整型
(float)( 5%3) 将 5%3的值转换成 float型
C程序设计(第三版) http://ccf.tsinghua.edu.cn 50
例 3.8 强制类型转换。
#include <stdio.h>
voidmain()
{float x;
int i;
x=3.6;
i=(int)x;
printf("x=%f,i=%d\n",x,i);
}
说明,有两种类型转换,一种是在运算时不必用户指定,系统自动进行的类型转换,如 3+6.5。
第二种是强制类型转换。当自动类型转换不能实现目的时,可以用强制类型转换。
运行结果:
x= 3.600000,i=3
C程序设计(第三版) http://ccf.tsinghua.edu.cn 51
3.8 算术运算符和算术表达式
(4) 自增、自减运算符作用是使变量的值增1或减1
如,
++i,--i(在使用i之前,先使i的值加
(减)1)
i++,i--(在使用i之后,使i的值加(
减)1)
C程序设计(第三版) http://ccf.tsinghua.edu.cn 52
3.8 算术运算符和算术表达式
i++与 ++i的区别:
++i是先执行i=i+1后,再使用i的值;
i++是先使用i的值后,再执行i=i+1。
例如:
①j=++i;
i的值先变成 4,再赋给j,j的值均为4
②j=i++;
先将 i的值 3赋给j,j的值为3,然后i变为4
C程序设计(第三版) http://ccf.tsinghua.edu.cn 53
3.8 算术运算符和算术表达式注意:
(1)自增运算符(++),自减运算符(--),
只能用于变量,而不能用于常量或表达式,
(2)++和--的结合方向是? 自右至左? 。
自增(减)运算符常用于循环语句中使循环变量自动加1。也用于指针变量,使指针指向下一个地址
C程序设计(第三版) http://ccf.tsinghua.edu.cn 54
3.8 算术运算符和算术表达式
(5) 有关表达式使用中的问题说明
① ANSI C并没有具体规定表达式中的子表达式的求值顺序,允许各编译系统自己安排。
例如,对表达式 a = f1( )+f2( )
并不是所有的编译系统都先调用 f1( ),然后调用 f2( )。在有的情况下结果可能不同。有时会出现一些令人容易搞混的问题,因此务必要小心谨慎 。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 55
3.8 算术运算符和算术表达式
② C语言中有的运算符为一个字符,有的运算符由两个字符组成,为避免误解,最好采取大家都能理解的写法。
例如,不要写成 i+++j的形式,而应写成
(i++)+j的形式
C程序设计(第三版) http://ccf.tsinghua.edu.cn 56
3.8 算术运算符和算术表达式
③ 在调用函数时,实参数的求值顺序,C标准并无统一规定。
例如,i的初值为3,如果有下面的函数调用:
printf( ″%d,%d ″,i,i++) 在有的系统中,从左至右求值,输出“3,3”。在多数系统中对函数参数的求值顺序是自右而左,pri
ntf函数输出的是“4,3”。以上这种写法不宜提倡,最好改写成
j = i++;
printf("%d,%d",j,i)
不要写出别人看不懂的也不知道系统会怎样执行程序
C程序设计(第三版) http://ccf.tsinghua.edu.cn 57
3.9 赋值运算符和赋值表达式
(1)赋值运算符赋值符号“=”就是赋值运算符,它的作用是将一个数据赋给一个变量。如“a=3”的作用是执行一次赋值操作(或称赋值运算)。把常量3赋给变量a。也可以将一个表达式的值赋给一个变量,
C程序设计(第三版) http://ccf.tsinghua.edu.cn 58
3.9 赋值运算符和赋值表达式
(2)类型转换如果赋值运算符两侧的类型不一致,但都是数值型或字符型时,在赋值时要进行类型转换 。
① 将浮点型数据 ( 包括单,双精度 ) 赋给整型变量时,舍弃浮点数的小数部分 。
如,i为整型变量,执行? i=3.56” 的结果是使
i的值为3,以整数形式存储在整型变量中,
C程序设计(第三版) http://ccf.tsinghua.edu.cn 59
3.9 赋值运算符和赋值表达式
② 将整型数据赋给单、双精度变量时,数值不变,
但以浮点数形式存储到变量中如,将 23赋给 float变量f,即执行f= 23,先将 23转换成 23,00000,再存储在f中。
将 23赋给 double型变量d,即执行d= 23,系统将 23补足有效位数字为 23,00000000000000,
然后以双精度浮点数形式存储到变量d中。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 60
3.9 赋值运算符和赋值表达式
③ 将一个 double型数据赋给 float变量时,截取其前面 7位有效数字,存放到 float变量的存储单元( 4个字节)中。但应注意数值范围不能溢出。
如,float f;double d=123.456789e100; f=d;
就出现溢出的错误。
如果将一个 float型数据赋给 double变量时,数值不变,有效位数扩展到 16位,在内存中以 8个字节存储
C程序设计(第三版) http://ccf.tsinghua.edu.cn 61
3.9 赋值运算符和赋值表达式
④ 字符型数据赋给整型变量时,由于字符只占 1个字节,而整型变量为2个字节,因此将字符数据(8个二进位)放到整型变量存储单元的低8位中。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 62
第一种情况,低位对应、高位补零若所用系统将字符处理为无符号字符类型,或程序已将字符变量定义为 unsigned char型,则将字符的8位放到整型变量低8位,高
8位补零。 例如:
将字符‘\ 376’赋给 int型变量i
第二种情况,符号扩展若所用系统(如 Turbo C++)将字符处理为带符号( signed
char)时,若字符最高位为0,则整型变量高8位补0;若字符最高位为1,则高8位全补1。这称为? 符号扩展?,这样做的目的是使数值保持不变,如变量c(字符 ‘ \ 376?)以整数形式输出为-2,i的值也是-2。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 63
3.9 赋值运算符和赋值表达式
⑤ 将一个 int,short,long型数据赋给一个 char型变量时,只将其低 8位原封不动地送到 char型变量(即截断)。 例如,int i=289;char c=′a′;c=i;
c的值为 33,
如果用,%c” 输出 c,将得到字符,!”
!的 ASCII码为 33。
赋值情况,
C程序设计(第三版) http://ccf.tsinghua.edu.cn 64
3.9 赋值运算符和赋值表达式
⑥ 将带符号的整型数据( int型)赋给 long型变量时,要进行符号扩展,将整型数的 16位送到
long型低 16位中,
如果 int型数据为正值(符号位为0),则 long
型变量的高 16位补0;
如果 int型变量为负值(符号位为1),则 long
型变量的高 16位补1,以保持数值不改变。
反之,若将一个 long型数据赋给一个 int型变量
,只将 long型数据中低 16位原封不动地送到整型变量(即截断)。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 65
3.9 赋值运算符和赋值表达式例如,int a; long b=8; a=b
赋值情况如下,
如果b= 65536(八进制数 0200000),
则赋值后a值为0。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 66
3.9 赋值运算符和赋值表达式
⑦ 将 unsigned int型数据赋给 long int型变量时,不存在符号扩展问题,只需将高位补0即可。将一个
unsigned类型数据赋给一个占字节数相同的非
unsigned型整型变量(例如,unsigned int
->int,unsigned long->long,unsigned short
->short),将 unsigned型变量的内容原样送到非
unsigned型变量中,但如果数据范围超过相应整型的范围,则会出现数据错误。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 67
3.9 赋值运算符和赋值表达式例如,unsigned int a= 65535; int b;b=a;
将a整个送到b中,由于b是 int,第1位是符号位,
因此 b成了负数。根据补码知识可知,b的值为 -1,
可以用 printf( ″%d″,b); 来验证。
⑧ 将非 unsigned型数据赋给长度相同的 unsigned型变量,也是原样照赋(符号位也作为数值一起传送)。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 68
例 3.9 有符号数据传送给无符号变量。
#include <stdio.h>
void main()
{ unsigned a;
int b=-1;
a=b;
print( ″ %u \n″,a);
}
说明:,%u” 是输出无符号数时所用的格式符。如果
b为正值,且在0~ 32767之间,则赋值后数值不变。
运行结果:
65535
C程序设计(第三版) http://ccf.tsinghua.edu.cn 69
3.9 赋值运算符和赋值表达式
(3) 复合的赋值运算符在赋值符,=,之前加上其他运算符,可以构成复合的运算符 。
例如,
a+=3 等价于 a=a+3
x *=y+8 等价于 x=x *(y+
8)
x%=3 等价于 x=x%3
以“a+=3”为例来说明,
它相当于使a进行一次自加
(3)的操作。即先使a加
3,再赋给a。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 70
3.9 赋值运算符和赋值表达式为便于记忆,可以这样理解:
① a += b (其中 a为变量,b为表达式 )
② a += b(将有下划线的? a+”移到? =”右侧
)
|___↑
③ a = a + b (在? =”左侧补上变量名 a)
C程序设计(第三版) http://ccf.tsinghua.edu.cn 71
3.9 赋值运算符和赋值表达式注意,如果b是包含若干项的表达式,则相当于它有括号。
如,① x %= y+3
② x %= (y+3)
|__↑
③ x = x %(y+3)(不要错写成 x=x%y+3)
凡是二元(二目)运算符,都可以与赋值符一起组合成复合赋值符。
C语言规定可以使用10种复合赋值运算符:
+=,-=,*=,/=,%=,<<=,>>=,&=
,∧ =,|=
C程序设计(第三版) http://ccf.tsinghua.edu.cn 72
3.9 赋值运算符和赋值表达式
(4) 赋值表达式由赋值运算符将一个变量和一个表达式连接起来的式子称为,赋值表达式,。
一般形式为,
<变量 ><赋值运算符 ><表达式 >
例如,,a= 5”是一个赋值表达式
C程序设计(第三版) http://ccf.tsinghua.edu.cn 73
3.9 赋值运算符和赋值表达式对赋值表达式求解的过程是:
①求赋值运算符右侧的?表达式?的值 ;
② 赋给赋值运算符左侧的变量。
例如,赋值表达式? a =3*5”的值为 15,执行表达式后,变量 a的值也是 15。
注意,一个表达式应该有一个值
C程序设计(第三版) http://ccf.tsinghua.edu.cn 74
3.9 赋值运算符和赋值表达式左值 (lvalue),赋值运算符左侧的标识符
变量可以作为左值 ;
而表达式就不能作为左值 (如 a+b);
常变量也不能作为左值,
右值 (lvalue):出现在赋值运算符右侧的表达式左值也可以出现在赋值运算符右侧,因而左值都可以作为右值。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 75
3.9 赋值运算符和赋值表达式赋值表达式中的“表达式”,又可以是一个赋值表达式。 例如,a=(b=5)
分析,括弧内的? b=5”是一个赋值表达式,它的值等于 5。执行表达式? a=(b=5)”相当于执行
b=5”和? a= b”两个赋值表达式。 赋值运算符按照? 自右而左? 的结合顺序,因此,? (b=
5)”
外面的括弧可以不要,即? a=(b=5)”和
a=b=5”
等价。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 76
3.9 赋值运算符和赋值表达式请分析下面的赋值表达式 ∶ (a=3*5)=4*3
分析,先执行括弧内的运算,将 15赋给 a,然后执行
4*3的运算,得 12,再把 12赋给 a。最后 a的值为 12,
整个表达式的值为 12。可以看到 ∶ (a=3*5)出现在赋值运算符的左侧,因此赋值表达式 (a=3*5)是左值注意,在对赋值表达式 (a=3*5)求解后,变量 a得到值
15执行 (a=3*5)=4*3时,实际上是将 4*3的积 12赋给变量 a,而不是赋给 3*5。
不能写成,
a=3*5=4*3
C程序设计(第三版) http://ccf.tsinghua.edu.cn 77
3.9 赋值运算符和赋值表达式赋值表达式也可以包含复合的赋值运算符。
如,a+=a-=a*a
分析,此赋值表达式的求解步骤如下 ∶
① 先进行? a-=a *a? 的运算,它相当于a=a-a *a,a的值为12- 144
=- 132。
②再进行? a+=-132? 的运算,相当于a =a+(-132),a的值为- 132-132= -
264。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 78
3.9 赋值运算符和赋值表达式将赋值表达式作为表达式的一种,使赋值操作不仅可以出现在赋值语句中,而且可以以表达式形式出现在其他语句(如输出语句、循环语句等)中。
如,printf("%d",a=b);
分析,如果 b的值为 3,则输出 a的值 (也是表达式 a=b的值 )为 3。在一个语句中完成了赋值和输出双重功能。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 79
3.10 逗号运算符和逗号表达式逗号运算符,将两个表达式连接起来,又称为? 顺序求值运算符?
如,3+ 5,6+8
一般形式,表达式1,表达式2
求解过程:
先求解表达式1,再求解表达式2。整个逗号表达式的值是表达式2的值。
逗号表达式 的值为 14
C程序设计(第三版) http://ccf.tsinghua.edu.cn 80
3.10 逗号运算符和逗号表达式例,逗号表达式 a=3 *5,a *4
分析,赋值运算符的优先级别高于逗号运算符,因此应先求解a=3 *5。
a的值为1 5,
然后求解a *4,
得60。整个逗号表达式的值为60。
一个逗号表达式又可以与另一个表达式组成一个新的逗号表达式如,(a=3 *5,a *4)
a+5先计算出a的值等于15,再进行a *4的运算得 60(但a值未变,仍为 15),再进行a
+5得20,即整个表达式的值为20。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 81
3.10 逗号运算符和逗号表达式逗号表达式的一般形式可以扩展为表达式1,表达式2,表达式3,……,表达式n
它的值为表达式n的值。
逗号运算符是所有运算符中级别最低的例,① x=(a=3,6 *3)
② x=a=3,6 *3
赋值表达式,
将一个逗号表达式的值赋给
x,x的值等于18
逗号表达式,包括一个赋值表达式和一个算术表达式,
x的值为3,整个逗号表达式的值为
18。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 82
3.10 逗号运算符和逗号表达式注意,并不是任何地方出现的逗号都是作为逗号运算符。例如函数参数也是用逗号来间隔的。
如,printf(“%d,%d,%d”,a,b,c);
“a,b,c”
并不是一个逗号表达式,它是 printf函数的 3
个参数
printf(“%d,%d,%d”,(a,b,c),b,c)“(a,b,
c)”是一个逗号表达式,
它的值等于c
的值。
本章要点
数据的描述规则
数据的操作规则
主要内容
3.1 C的数据类型
3.2 常量与变量
3.3 整型数据
3.4 浮点型数据运行
3.5 字符型数据
主要内容
3.6变量赋初值
3.7 各类数值型数据间的混合运算
3.8 算术运算符和算术表达式
3.9 赋值运算符和赋值表达式
3.10 逗号运算符和逗号表达式
C程序设计(第三版) http://ccf.tsinghua.edu.cn 5
3.1 C的数据类型
C语言提供了以下一些数据类型。
数据类型 构造类型指针类型空类型( 无值类型) void
枚举类型 enum
数组类型结构类型 struct
联合类型 union
基本类型整型 int
字符型 char
实型(浮点型)
单精度实型
float
双精度实型
double
C程序设计(第三版) http://ccf.tsinghua.edu.cn 6
3.2 常量与变量
3.2.1 常量和符号常量
在程序运行过程中,其值不能被改变的量称为常量
常量区分为不同的类型:
整型 100,125,-100,0
实型 3.14,0.125,-3.789
字符型 ‘ a?,?b?,‘ 2?
字符串 ‘ a?,?ab?,‘ 1232?
C程序设计(第三版) http://ccf.tsinghua.edu.cn 7
例 3.1 符号常量的使用
#define PRICE 30
#include <stdio.h>
void main ( )
{
int num,total;
num=10;
total=num * PRICE;
printf(″total=%d \n″,total);
} 符号常量,用一个标识符代表一个常量。
符号常量的值在其作用域内不能改变,
也不能再被赋值。
运行结果,total=300
说明,程序中用 #define命令行定义 PRICE
代表常量 30,此后凡在本文件中出现的
PRICE都代表 30,可以和常量一样进行运算说明,如再用赋值语句给 PRICE赋值是错的
PRICE=40; /* 错误,不能给符号常量赋值
C程序设计(第三版) http://ccf.tsinghua.edu.cn 8
3.2 常量与变量
3.2.2 变量
变量代表内存中具有特定属性的一个存储单元,它用来存放数据,这就是变量的值,在程序运行期间,这些值是可以改变的。
变量名实际上是一个以一个名字对应代表一个地址,在对程序编译连接时由编译系统给每一个变量名分配对应的内存地址。从变量中取值,实际上是通过变量名找到相应的内存地址,从该存储单元中读取数据。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 9
3.2 常量与变量变量命名的规定,C语言规定标识符只能由字母、数字和下划线三种字符组成,且第一个字符必须为字母或下划线。
例,sum,_total,month,Student_name,
lotus_1_2_3,BASIC,li_ling
M.D.John,¥ 123,3D64,a>b
C程序设计(第三版) http://ccf.tsinghua.edu.cn 10
3.2 常量与变量注意:
编译系统将大写字母和小写字母认为是两个不同的字符。
建议变量名的长度最好不要超过 8个字符。
在选择变量名和其它标识符时,应注意做到? 见名知意?,即选有含意的英文单词
(或其缩写)作标识符。
要求对所有用到的变量作强制定义,也就是? 先定义,后使用? 。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 11
3.3 整型数据
3.3.1整型常量的表示方法整型常量即整常数。在C语言中,整常数可用以下三种形式表示:
(1)十进制整数。
如,123,-456.4。
(2)八进制整数。以 0头的数是八进制数。
如,0123表示八进制数 123,等于十进制数
83,-011表示八进制数 -11,即十进制数 -9。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 12
3.3 整型数据
(3)十六进制整数。以 0x开头的数是 16进制数。
如,0x123,代表 16进制数 123,等于十进制数
291。 -0x12等于十进制数- 10。
3.3.2 整型变量
(1)整型数据在内存中的存放形式数据在内存中是以二进制形式存放的。
如,int i; /* 定义为整型变量 */
i=10; /* 给 i赋以整数 10 */
C程序设计(第三版) http://ccf.tsinghua.edu.cn 13
3.3 整型数据注意:
十进制数 10的二进制形式为 1010,Turbo
C 2.0和 Turbo C++ 3.0为一个整型变量在内存中分配 2个字节的存储单元 (不同的编译系统为整型数据分配的字节数是不相同的,VC++ 6.0则分配 4个字节)。
数值是以补码 (complement) 表示的。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 14
3.3 整型数据
(2)整型变量的分类共六种有符号基本整型有符号短整型有符号长整型无符号基本整型无符号短整型无符号长整型
(signed)int
(signed)short (int )
(signed) long (int)
unsigned int
unsigned short (int)
unsigned long (int)
注意,括号表示其中的内容是可选的,
C程序设计(第三版) http://ccf.tsinghua.edu.cn 15
3.3 整型数据
整数类型的有关数据:
类型 类型说明符 长度 数的范围基本型 int 2字节 -32768~ 32767
短整型 short 2字节 -215~ 215-1
长整型 long 4字节 -231~ 231-1
无符号整型 unsigned 2字节 0~ 65535
无符号短整型 unsigned short 2字节 0~ 65535
无符号长整型 unsigned long 4字节 0~( 232-1)
C程序设计(第三版) http://ccf.tsinghua.edu.cn 16
3.3 整型数据例如:整数 13在内存中实际存放的情况:
C程序设计(第三版) http://ccf.tsinghua.edu.cn 17
3.3 整型数据
(3)整型变量的定义,
C规定在程序中所有用到的变量都必须在程序中定义,即? 强制类型定义? 。
例如,
int a,b(指定变量a、b为整型)
unsigned short c,d;(指定变量c、d为无符号短整型)
long e,f;(指定变量e、f为长整型)
C程序设计(第三版) http://ccf.tsinghua.edu.cn 18
例 3.2 整型变量的定义与使用
#include <stdio.h>
void main()
{ int a,b,c,d; /* 指定a、b、c、d为整
*/
unsigned u; / *
*/
a= 12;b= -24;u= 10;
c=a+u;d=b+u;
printf( ″a+u=%d,b+u=%d\n ″
,c,d);
}
说明,可以看到不同种类的整型数据可以进行算术运算运行结果,a+u=22
,b+u=-14
C程序设计(第三版) http://ccf.tsinghua.edu.cn 19
例 3.3 整型数据的溢出
#include <stdio.h>
void main()
{int a,b;
a=32767;
b=a+1;
printf(“%d,%d \n”,a,b);
} 说明,数值是以补码表示的。一个整型变量只能容纳 -32768~ 32767范围内的数,无法表示大于 32767或小于 -32768的数。遇此情况就发生“溢出”。
运行结果,32767,-32768
C程序设计(第三版) http://ccf.tsinghua.edu.cn 20
3.3 整型数据
3.3.3 整型常量的类型
(1)一个整数,如果其值在 -32768~ +32767范围内,认为它是 int型,它可以赋值给 int型和 long int型变量。
(2) 一个整数,如果其值超过了上述范围,
而在 -2147483637~ +2147483647范围内,则认为它是为长整型。可以将它赋值给一个
long int型变量。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 21
3.3 整型数据
(3) 如果所用的 C版本(如 Turbo C)分配给
short int与 int型数据在内存中占据的长度相同,则它的表数范围与 int型相同。因此一个 int型的常量同时也是一个 short int型常量,可以赋给 int型或 short int型变量。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 22
3.3 整型数据
(4) 一个整常量后面加一个字母 u或 U,认为是 unsigned int型,如 12345u,在内存中按 unsigned int规定的方式存放 (存储单元中最高位不作为符号位,而用来存储数据 )。如果写成 -12345u,则先将 -12345
转换成其补码 53191,然后按无符号数存储。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 23
3.3 整型数据
(5) 在一个整常量后面加一个字母 l或 L
,则认为是 long int型常量。
例如,123l.432L.0L
用于函数调用中。
如果函数的形参为 long int型,则要求实参也为 long int型。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 24
3.4 浮点型数据
3.4.1浮点型常量的表示方法两种表示形式小数指数
0.123
3e-3
注意,字母 e(或 E)之前必须有数字,且 e后面的指数必须为整数
1e3,1.8e-3,-123e-6,-.1e-3
e3,2.1e3.5,.e3,e
C程序设计(第三版) http://ccf.tsinghua.edu.cn 25
3.4 浮点型数据规范化的指数形式:
在字母 e(或 E)之前的小数部分中,小数点左边应有一位(且只能有一位)非零的数字,
例如,123.456可以表示为:
123.456e0,12.3456e1,1.23456e2,
0.123456e3,0.0123456e4,0.00123456e
其中的 1.23456e3称为? 规范化的指数形式
。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 26
3.4 浮点型数据
3.4.2 浮点型变量
(1)浮点型数据在内存中的存放形式一个浮点型数据一般在内存中占 4
个字节 (32位 )。与整型数据的存储方式不同,浮点型数据是按照指数形式存储的。系统把一个浮点型数据分成小数部分和指数部分,分别存放。指数部分采用规范化的指数形式。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 27
类型 位数 数的范围 有效数字
float 32 10-37 ~ 1038 6~7 位
double型 64 10-307~10308 15~16位
long double 128 10-4931~104932 18~19位
3.4 浮点型数据
(2) 浮点型变量的分类浮点型变量分为单精度( float型)、双精度
( double型)和长双精度型( long double)三类形式。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 28
例 3.4 浮点型数据的舍入误差
#include <stdio.h>
void main()
{float a,b;
a = 123456.789e5;
b = a + 20 ;
printf(“%f \n”,b);
}说明,一个浮点型变量只能保证的有效数字是 7位有效数字,后面的数字是无意义的,并不准确地表示该数。应当避免将一个很大的数和一个很小的数直接相加或相减,否则就会“丢失”小的数运行结果,123456.789e5
C程序设计(第三版) http://ccf.tsinghua.edu.cn 29
3.4 浮点型数据
3.4.3 浮点型常量的类型
C编译系统将浮点型常量作为双精度来处理。
例如,f = 2.45678 * 4523.65
系统先把 2.45678和 4523.65作为双精度数
,然后进行相乘的运算,得到的乘也是一个双精度数。最后取其前 7位赋给浮点型变量 f
。如是在数的后面加字母 f或 F(如 1.65f,
654.87F),这样编译系统就会把它们按单精度( 32位)处理。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 30
3.5 字符型数据
3.5.1 字符常量
(1)用单引号包含的一个字符是字符型常量
(2)只能包含一个字符
‘a’,’A’,‘1’
‘abc’,,a”?
C程序设计(第三版) http://ccf.tsinghua.edu.cn 31
3.5 字符型数据有些以? \”开头的特殊字符称为转义字符
\n 换行
\t 横 向跳格
\r 回 车
\\ 反 斜杠
\ddd ddd表示 1到 3位八进制数字
\xhh hh表示 1到 2位十六进制数字
C程序设计(第三版) http://ccf.tsinghua.edu.cn 32
例 3.5 转义字符的使用
#include <stdio.h>
void main()
{printf( ″ ab c \t
de\rf\tg\n″ ) ;
printf(″h \ti\b\bj k\n″);
}
打印机上的显示结果:
fab c gde
h jik
显示屏上 的运行结果:
f gde
h k
C程序设计(第三版) http://ccf.tsinghua.edu.cn 33
3.5 字符型数据
3.5.2字符变量
字符型变量用来存放字符常量,注意只能放一个字符 。
字符变量的定义形式如下,char c1,c2;
在本函数中可以用下面语句对 c1,c2赋值:
c1= ‘ a?; c2= ‘ b? ;
一个字符变量在内存中占一个字节。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 34
3.5 字符型数据
3.5.3 字符数据在内存中的存储形式及其使用方法
一个字符常量存放到一个字符变量中,实际上并不是把该字符的字型放到内存中去,而是将该字符的相应的 ASCII代码放到存储单元中。这样使字符型数据和整型数据之间可以通用。
注意,
一个字符数据既可以以字符形式输出,也可以以整数形式输出。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 35
例 3.6 向字符变量赋以整数。
#include <stdio.h>
void main()
{char c1,c2;
c1=97;
c2=98;
printf(“%c %c \n”,c1,c2);
printf(“%d %d \n”,c1,c2);
}? 说明,在第3和第 4行中,将整数 97和 98分别赋给 c1和 c2,它的作用相当于以下两个赋值语句:
c1= ′ a ′;c2 = ′ b ′;
因为’ a’ 和’ b’ 的 ASCII码为 97和 98
运行结果:
a b
97 98
C程序设计(第三版) http://ccf.tsinghua.edu.cn 36
例 3.7 大小写字母的转换
#include <stdio.h>
void main()
{char c1,c2;
c1=’a’;
c2=’b’;
c1=c1-32;
c2=c2-32;
printf(,% c % c″,c1,c2);
}
说明,程序的作用是将两个小写字母 a和 b转换成大写字母 A和 B。从ASCII代码表中可以看到每一个小写字母比它相应的大写字母的 ASCII码大 32。C语言允许字符数据与整数直接进行算术运算。
运行结果,A B
C程序设计(第三版) http://ccf.tsinghua.edu.cn 37
3.5 字符型数据说明:
有些系统(如 Turbo C)将字符变量定义为 signed
char型。其存储单元中的最高位作为符号位,它的取值范围是 -128~ 127。如果在字符变量中存放一个
ASCII码为 0~ 127间的字符,由于字节中最高位为 0,
因此用 %d输出字符变量时,输出的是一个正整数。如果在字符变量中存放一个 ASCII码为 128~ 255间的字符,由于在字节中最高位为 1,用 %d格式符输出时,
就会得到一个负整数。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 38
3.5 字符型数据
3.5.4字符串常量
字符串常量是一对双撇号括起来的字符序列,
合法的字符串常量,
How do you do.”,“CHINA”,? a”,
$123.45”
可以输出一个字符串,如
printf(“How do you do.”);
C程序设计(第三版) http://ccf.tsinghua.edu.cn 39
3.5 字符型数据
a ’ 是字符常量,? a? 是字符串常量,二者不同。
如,假设C被指定为字符变量,char cc=’a’;
c=” a” ; c=” CHINA”;
结论,不能把一个字符串常量赋给一个字符变量。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 40
C规定:在每一个字符串常量的结尾加一个? 字符串结束标志?,以便系统据此判断字符串是否结束。
C规定以字符 ’ \0 ’ 作为字符串结束标志。如,如果有一个字符串常量”CHINA”,实际上在内存中是,C H I N A \0
它占内存单元不是5个字符,而是6个字符,
最后一个字符为’\0’。但在输出时不输出’
\0’。
3.5 字符型数据
C程序设计(第三版) http://ccf.tsinghua.edu.cn 41
3.6 变量赋初值字符串常量
(1)C语言允许在定义变量的同时使变量初始化如,int a=3; // 指定a为整型变量,初值为3
float f=3.56; // 指定f为浮点型变量,初值为3,56
char c= ‘a’; // 指定c为字符变量,初值为‘ a’
C程序设计(第三版) http://ccf.tsinghua.edu.cn 42
3.6 变量赋初值
(2)可以使被定义的变量的一部分赋初值。
如,int a,b,c=5; 表示指定a、b、c为整型变量,但只对c初始化,c的初值为5
(3)如果对几个变量赋以同一个初值,
应写成,int a=3,b=3,c=3; 表示a、b、c
的初值都是3。
不能写成 ∶ int a=b=c3;
注意,初始化不是在编译阶段完成的而是在程序运行时执行本函数时赋初值的,相当于有一个赋值语句。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 43
3.7 各类数值型数据间的混合运算混合运算,整型(包括 int,short,long)、浮点型(包括 float,double)可以混合运算。
在进行运算时,不同类型的数据要先转换成同一类型,然后进行运算,
说明:
这种类型转换是由系统自动进行的。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 44
3.8 术运算符和算术表达式
3.8.1 C运算符简介
C的运算符有以下几类:
(1)算术运算符 ( + - * / %)
(2)关系运算符 (><==>=<=!=)
(3)逻辑运算符 (!&& ||)
(4)位运算符 ( << >> ~ |∧ &)
(5)赋值运算符 (=及其扩展赋值运算符)
(6)条件运算符 (?:)
(7)逗号运算符 (,)
C程序设计(第三版) http://ccf.tsinghua.edu.cn 45
3.8 算术运算符和算术表达式
(8)指针运算符 ( *和&)
(9)求字节数运算符(sizeof)
(10)强制类型转换运算符( (类型) )
(11)分量运算符(,->)
(12)下标运算符([ ])
(13)其他 (如函数调用运算符())
C程序设计(第三版) http://ccf.tsinghua.edu.cn 46
3.8 算术运算符和算术表达式
3.8.2 算术运算符和算术表达式
(1)基本的算术运算符,
+ ( 加法运算符,或正值运算符。如,3+5、+3)
- ( 减法运算符,或负值运算符。如,5-2、-3)
* (乘法运算符。如,3 *5)
/ ( 除法运算符。如,5/3)
% ( 模运算符,或称求余运算符,%两侧均应为整型数据,如,7%4的值为3)。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 47
3.8 算术运算符和算术表达式
(2) 算术表达式和运算符的优先级与结合性基本的算术运算符,
用算术运算符和括号将运算对象 (也称操作数)
连接起来的、符合C语法规则的式子,称为C算术表达式。运算对象包括常量、变量、函数等。
例如,a *b/c-1,5+ ′a′
是一个合法的表达式
C程序设计(第三版) http://ccf.tsinghua.edu.cn 48
C语言规定了运算符的优先级和结合性。
在表达式求值时,先按运算符的优先级别高低次序执行,例如先乘除后加减。
C规定了各种运算符的结合方向 (结合性 )
算术运算符的结合方向为“自左至右”,即先左后右 。
3.8 算术运算符和算术表达式
C程序设计(第三版) http://ccf.tsinghua.edu.cn 49
3.8 算术运算符和算术表达式
(3)强制类型转换运算符可以利用强制类型转换运算符将一个表达式转换成所需类型。
一般形式,(类型名)(表达式)
例如:
(double)a 将a转换成 double类型
(int)(x+y) 将 x+y的值转换成整型
(float)( 5%3) 将 5%3的值转换成 float型
C程序设计(第三版) http://ccf.tsinghua.edu.cn 50
例 3.8 强制类型转换。
#include <stdio.h>
voidmain()
{float x;
int i;
x=3.6;
i=(int)x;
printf("x=%f,i=%d\n",x,i);
}
说明,有两种类型转换,一种是在运算时不必用户指定,系统自动进行的类型转换,如 3+6.5。
第二种是强制类型转换。当自动类型转换不能实现目的时,可以用强制类型转换。
运行结果:
x= 3.600000,i=3
C程序设计(第三版) http://ccf.tsinghua.edu.cn 51
3.8 算术运算符和算术表达式
(4) 自增、自减运算符作用是使变量的值增1或减1
如,
++i,--i(在使用i之前,先使i的值加
(减)1)
i++,i--(在使用i之后,使i的值加(
减)1)
C程序设计(第三版) http://ccf.tsinghua.edu.cn 52
3.8 算术运算符和算术表达式
i++与 ++i的区别:
++i是先执行i=i+1后,再使用i的值;
i++是先使用i的值后,再执行i=i+1。
例如:
①j=++i;
i的值先变成 4,再赋给j,j的值均为4
②j=i++;
先将 i的值 3赋给j,j的值为3,然后i变为4
C程序设计(第三版) http://ccf.tsinghua.edu.cn 53
3.8 算术运算符和算术表达式注意:
(1)自增运算符(++),自减运算符(--),
只能用于变量,而不能用于常量或表达式,
(2)++和--的结合方向是? 自右至左? 。
自增(减)运算符常用于循环语句中使循环变量自动加1。也用于指针变量,使指针指向下一个地址
C程序设计(第三版) http://ccf.tsinghua.edu.cn 54
3.8 算术运算符和算术表达式
(5) 有关表达式使用中的问题说明
① ANSI C并没有具体规定表达式中的子表达式的求值顺序,允许各编译系统自己安排。
例如,对表达式 a = f1( )+f2( )
并不是所有的编译系统都先调用 f1( ),然后调用 f2( )。在有的情况下结果可能不同。有时会出现一些令人容易搞混的问题,因此务必要小心谨慎 。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 55
3.8 算术运算符和算术表达式
② C语言中有的运算符为一个字符,有的运算符由两个字符组成,为避免误解,最好采取大家都能理解的写法。
例如,不要写成 i+++j的形式,而应写成
(i++)+j的形式
C程序设计(第三版) http://ccf.tsinghua.edu.cn 56
3.8 算术运算符和算术表达式
③ 在调用函数时,实参数的求值顺序,C标准并无统一规定。
例如,i的初值为3,如果有下面的函数调用:
printf( ″%d,%d ″,i,i++) 在有的系统中,从左至右求值,输出“3,3”。在多数系统中对函数参数的求值顺序是自右而左,pri
ntf函数输出的是“4,3”。以上这种写法不宜提倡,最好改写成
j = i++;
printf("%d,%d",j,i)
不要写出别人看不懂的也不知道系统会怎样执行程序
C程序设计(第三版) http://ccf.tsinghua.edu.cn 57
3.9 赋值运算符和赋值表达式
(1)赋值运算符赋值符号“=”就是赋值运算符,它的作用是将一个数据赋给一个变量。如“a=3”的作用是执行一次赋值操作(或称赋值运算)。把常量3赋给变量a。也可以将一个表达式的值赋给一个变量,
C程序设计(第三版) http://ccf.tsinghua.edu.cn 58
3.9 赋值运算符和赋值表达式
(2)类型转换如果赋值运算符两侧的类型不一致,但都是数值型或字符型时,在赋值时要进行类型转换 。
① 将浮点型数据 ( 包括单,双精度 ) 赋给整型变量时,舍弃浮点数的小数部分 。
如,i为整型变量,执行? i=3.56” 的结果是使
i的值为3,以整数形式存储在整型变量中,
C程序设计(第三版) http://ccf.tsinghua.edu.cn 59
3.9 赋值运算符和赋值表达式
② 将整型数据赋给单、双精度变量时,数值不变,
但以浮点数形式存储到变量中如,将 23赋给 float变量f,即执行f= 23,先将 23转换成 23,00000,再存储在f中。
将 23赋给 double型变量d,即执行d= 23,系统将 23补足有效位数字为 23,00000000000000,
然后以双精度浮点数形式存储到变量d中。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 60
3.9 赋值运算符和赋值表达式
③ 将一个 double型数据赋给 float变量时,截取其前面 7位有效数字,存放到 float变量的存储单元( 4个字节)中。但应注意数值范围不能溢出。
如,float f;double d=123.456789e100; f=d;
就出现溢出的错误。
如果将一个 float型数据赋给 double变量时,数值不变,有效位数扩展到 16位,在内存中以 8个字节存储
C程序设计(第三版) http://ccf.tsinghua.edu.cn 61
3.9 赋值运算符和赋值表达式
④ 字符型数据赋给整型变量时,由于字符只占 1个字节,而整型变量为2个字节,因此将字符数据(8个二进位)放到整型变量存储单元的低8位中。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 62
第一种情况,低位对应、高位补零若所用系统将字符处理为无符号字符类型,或程序已将字符变量定义为 unsigned char型,则将字符的8位放到整型变量低8位,高
8位补零。 例如:
将字符‘\ 376’赋给 int型变量i
第二种情况,符号扩展若所用系统(如 Turbo C++)将字符处理为带符号( signed
char)时,若字符最高位为0,则整型变量高8位补0;若字符最高位为1,则高8位全补1。这称为? 符号扩展?,这样做的目的是使数值保持不变,如变量c(字符 ‘ \ 376?)以整数形式输出为-2,i的值也是-2。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 63
3.9 赋值运算符和赋值表达式
⑤ 将一个 int,short,long型数据赋给一个 char型变量时,只将其低 8位原封不动地送到 char型变量(即截断)。 例如,int i=289;char c=′a′;c=i;
c的值为 33,
如果用,%c” 输出 c,将得到字符,!”
!的 ASCII码为 33。
赋值情况,
C程序设计(第三版) http://ccf.tsinghua.edu.cn 64
3.9 赋值运算符和赋值表达式
⑥ 将带符号的整型数据( int型)赋给 long型变量时,要进行符号扩展,将整型数的 16位送到
long型低 16位中,
如果 int型数据为正值(符号位为0),则 long
型变量的高 16位补0;
如果 int型变量为负值(符号位为1),则 long
型变量的高 16位补1,以保持数值不改变。
反之,若将一个 long型数据赋给一个 int型变量
,只将 long型数据中低 16位原封不动地送到整型变量(即截断)。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 65
3.9 赋值运算符和赋值表达式例如,int a; long b=8; a=b
赋值情况如下,
如果b= 65536(八进制数 0200000),
则赋值后a值为0。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 66
3.9 赋值运算符和赋值表达式
⑦ 将 unsigned int型数据赋给 long int型变量时,不存在符号扩展问题,只需将高位补0即可。将一个
unsigned类型数据赋给一个占字节数相同的非
unsigned型整型变量(例如,unsigned int
->int,unsigned long->long,unsigned short
->short),将 unsigned型变量的内容原样送到非
unsigned型变量中,但如果数据范围超过相应整型的范围,则会出现数据错误。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 67
3.9 赋值运算符和赋值表达式例如,unsigned int a= 65535; int b;b=a;
将a整个送到b中,由于b是 int,第1位是符号位,
因此 b成了负数。根据补码知识可知,b的值为 -1,
可以用 printf( ″%d″,b); 来验证。
⑧ 将非 unsigned型数据赋给长度相同的 unsigned型变量,也是原样照赋(符号位也作为数值一起传送)。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 68
例 3.9 有符号数据传送给无符号变量。
#include <stdio.h>
void main()
{ unsigned a;
int b=-1;
a=b;
print( ″ %u \n″,a);
}
说明:,%u” 是输出无符号数时所用的格式符。如果
b为正值,且在0~ 32767之间,则赋值后数值不变。
运行结果:
65535
C程序设计(第三版) http://ccf.tsinghua.edu.cn 69
3.9 赋值运算符和赋值表达式
(3) 复合的赋值运算符在赋值符,=,之前加上其他运算符,可以构成复合的运算符 。
例如,
a+=3 等价于 a=a+3
x *=y+8 等价于 x=x *(y+
8)
x%=3 等价于 x=x%3
以“a+=3”为例来说明,
它相当于使a进行一次自加
(3)的操作。即先使a加
3,再赋给a。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 70
3.9 赋值运算符和赋值表达式为便于记忆,可以这样理解:
① a += b (其中 a为变量,b为表达式 )
② a += b(将有下划线的? a+”移到? =”右侧
)
|___↑
③ a = a + b (在? =”左侧补上变量名 a)
C程序设计(第三版) http://ccf.tsinghua.edu.cn 71
3.9 赋值运算符和赋值表达式注意,如果b是包含若干项的表达式,则相当于它有括号。
如,① x %= y+3
② x %= (y+3)
|__↑
③ x = x %(y+3)(不要错写成 x=x%y+3)
凡是二元(二目)运算符,都可以与赋值符一起组合成复合赋值符。
C语言规定可以使用10种复合赋值运算符:
+=,-=,*=,/=,%=,<<=,>>=,&=
,∧ =,|=
C程序设计(第三版) http://ccf.tsinghua.edu.cn 72
3.9 赋值运算符和赋值表达式
(4) 赋值表达式由赋值运算符将一个变量和一个表达式连接起来的式子称为,赋值表达式,。
一般形式为,
<变量 ><赋值运算符 ><表达式 >
例如,,a= 5”是一个赋值表达式
C程序设计(第三版) http://ccf.tsinghua.edu.cn 73
3.9 赋值运算符和赋值表达式对赋值表达式求解的过程是:
①求赋值运算符右侧的?表达式?的值 ;
② 赋给赋值运算符左侧的变量。
例如,赋值表达式? a =3*5”的值为 15,执行表达式后,变量 a的值也是 15。
注意,一个表达式应该有一个值
C程序设计(第三版) http://ccf.tsinghua.edu.cn 74
3.9 赋值运算符和赋值表达式左值 (lvalue),赋值运算符左侧的标识符
变量可以作为左值 ;
而表达式就不能作为左值 (如 a+b);
常变量也不能作为左值,
右值 (lvalue):出现在赋值运算符右侧的表达式左值也可以出现在赋值运算符右侧,因而左值都可以作为右值。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 75
3.9 赋值运算符和赋值表达式赋值表达式中的“表达式”,又可以是一个赋值表达式。 例如,a=(b=5)
分析,括弧内的? b=5”是一个赋值表达式,它的值等于 5。执行表达式? a=(b=5)”相当于执行
b=5”和? a= b”两个赋值表达式。 赋值运算符按照? 自右而左? 的结合顺序,因此,? (b=
5)”
外面的括弧可以不要,即? a=(b=5)”和
a=b=5”
等价。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 76
3.9 赋值运算符和赋值表达式请分析下面的赋值表达式 ∶ (a=3*5)=4*3
分析,先执行括弧内的运算,将 15赋给 a,然后执行
4*3的运算,得 12,再把 12赋给 a。最后 a的值为 12,
整个表达式的值为 12。可以看到 ∶ (a=3*5)出现在赋值运算符的左侧,因此赋值表达式 (a=3*5)是左值注意,在对赋值表达式 (a=3*5)求解后,变量 a得到值
15执行 (a=3*5)=4*3时,实际上是将 4*3的积 12赋给变量 a,而不是赋给 3*5。
不能写成,
a=3*5=4*3
C程序设计(第三版) http://ccf.tsinghua.edu.cn 77
3.9 赋值运算符和赋值表达式赋值表达式也可以包含复合的赋值运算符。
如,a+=a-=a*a
分析,此赋值表达式的求解步骤如下 ∶
① 先进行? a-=a *a? 的运算,它相当于a=a-a *a,a的值为12- 144
=- 132。
②再进行? a+=-132? 的运算,相当于a =a+(-132),a的值为- 132-132= -
264。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 78
3.9 赋值运算符和赋值表达式将赋值表达式作为表达式的一种,使赋值操作不仅可以出现在赋值语句中,而且可以以表达式形式出现在其他语句(如输出语句、循环语句等)中。
如,printf("%d",a=b);
分析,如果 b的值为 3,则输出 a的值 (也是表达式 a=b的值 )为 3。在一个语句中完成了赋值和输出双重功能。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 79
3.10 逗号运算符和逗号表达式逗号运算符,将两个表达式连接起来,又称为? 顺序求值运算符?
如,3+ 5,6+8
一般形式,表达式1,表达式2
求解过程:
先求解表达式1,再求解表达式2。整个逗号表达式的值是表达式2的值。
逗号表达式 的值为 14
C程序设计(第三版) http://ccf.tsinghua.edu.cn 80
3.10 逗号运算符和逗号表达式例,逗号表达式 a=3 *5,a *4
分析,赋值运算符的优先级别高于逗号运算符,因此应先求解a=3 *5。
a的值为1 5,
然后求解a *4,
得60。整个逗号表达式的值为60。
一个逗号表达式又可以与另一个表达式组成一个新的逗号表达式如,(a=3 *5,a *4)
a+5先计算出a的值等于15,再进行a *4的运算得 60(但a值未变,仍为 15),再进行a
+5得20,即整个表达式的值为20。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 81
3.10 逗号运算符和逗号表达式逗号表达式的一般形式可以扩展为表达式1,表达式2,表达式3,……,表达式n
它的值为表达式n的值。
逗号运算符是所有运算符中级别最低的例,① x=(a=3,6 *3)
② x=a=3,6 *3
赋值表达式,
将一个逗号表达式的值赋给
x,x的值等于18
逗号表达式,包括一个赋值表达式和一个算术表达式,
x的值为3,整个逗号表达式的值为
18。
C程序设计(第三版) http://ccf.tsinghua.edu.cn 82
3.10 逗号运算符和逗号表达式注意,并不是任何地方出现的逗号都是作为逗号运算符。例如函数参数也是用逗号来间隔的。
如,printf(“%d,%d,%d”,a,b,c);
“a,b,c”
并不是一个逗号表达式,它是 printf函数的 3
个参数
printf(“%d,%d,%d”,(a,b,c),b,c)“(a,b,
c)”是一个逗号表达式,
它的值等于c
的值。