2.1 数据和运算

2.1.1 数的进制

我们在日常生活中大多使用十进制,即逢10进1,这主要是由人们的使用习惯决定的。其实在生活中还有许多其他不同的进位制度,如时间的表示方法是六十进制,即1小时等于60分,1分钟等于60秒等。还有常用的表示数量的单位“一打”是十二进制等。在计算机中,常用的进位制度有二进制、十进制、八进制和十六进制。

1.二进制(Binary)

二进制数由0和1两个符号来表示,基数为2,按逢2进1、借1算2的规则计数。例如:

10110011 1010 1111000010101101

2.十进制(Decimal)

十进制数由0、1、2、3、4、5、6、7、8、9、0这10个数字符号表示,基数为10,按逢10进1、借1算10的规则计数。例如:

128 23 47

3.八进制(Octal)

八进制数由0、1、2、3、4、5、6、7这8个数字符号表示,基数为8,按逢8进1、借1算8的规则计数。例如:

70 360 777

4.十六进制(Hexadecimal)

十六进制数由0、1、2、3、4、5、6、7、8、9、A、B、C、D、E、F这16个数字符号表示,基数为16,按逢16进1、借1算16的规则计数。在C语言中表示十六进制数时,大小写字母的含义相同。例如:

EF08 5ac 7BF

对于不同进制的数字间的转换在这里不做太多的叙述,对于初学者来讲,使用PC中的计算器来进行不同进制数字间的转换是一个方便快捷的办法,如图2-1所示。打开计算器软件,如要将十进制数“254”转换成二进制数,首先在计算器中选择十进制,输入数字“254”,再用鼠标单击选择二进制,这时计算器中即可显示经转换后的二进制数“11111110”。

图2-1 使用计算器进行进制转换

2.1.2 码制

在计算机内部,所有的信息都要使用二进制的方法来表示,因为二进制的0和1两个数字恰好与存储单元的“有”和“无”相对应。不仅如此,数的正负符号“+”或“-”也需要用二进制数来表示,在通常情况下,用0表示正数的符号“+”,用1表示负数的符号“-”。当数的符号和数值表示方法使用二进制时,这样的数被称为“机器码”。机器码有不同的码制,对应不同的表示方法,常用的码制有原码、反码和补码3种。

·原码:原码用最高位表示数的符号位,数值部分用二进制的绝对值表示。

·反码:正数的反码与其原码相同,负数的反码是将符号位除外,其他各位按位取反。

·补码:正数的补码与其原码相同,负数的补码是将其取反码后加1。

数的原码、反码和补码的表示方法详见表2-1。

表2-1 数的原码、反码和补码

2.1.3 数据类型

程序运行的目的是对数据进行处理,在C语言中数据是有类型区分的,具体分类如下:

在以上的数据类型分类中,基本类型是不可以再次拆分为其他数据类型的;构造类型则是在基本类型的基础上,按照一定方式组合而成的数据类型;指针类型是一种特殊的数据类型,其值通常用来表示某一个量在内存中的存放地址;空类型是指在对函数进行定义时,如函数没有返回值,就在函数的名称前面加上void,以此表明该函数是“空类型”。

2.1.4 常量

常量是指在程序的运行过程中,其值不能被改变的量。常量的种类有整型、实型、字符型和字符串4种。

1)整型常量:十进制的整数表示方法非常简单,如:29、–18、156等。十六进制的整数通常以0x(或0X)开头,如:0xFE、0xD7A9、0x7D等。八进制的整数则以0开头,如:057(其值相当于十进制的47)。

2)实型常量:实数有两种表示方法,一种是十进制的小数形式,如:0.625、–16.5等;另一种是采用指数形式,即用e(或E)后面跟一个整数,表示以10为底的幂指数,如:256.5的表示方法是2.565e2。

3)字符型常量:字符型常量的表示方法是用单引号引出,如‘a’‘B’等。

4)字符串常量:字符串常量用双引号引出,如“GOOD”“thank you”等。

2.1.5 变量

变量是指在程序运行过程中其值可以改变的量。在C语言中使用变量时,要先给变量命名,还要给变量定义数据类型,有时还需指定变量的存储地点。

为什么要给变量定义数据类型呢?在数学上,一个数可以是+∞也可以是–∞,但是在计算机中,存储单元是有限的,因此必须根据数据的大小为其分配合适的存储空间。定义数据类型实际上就是为变量在内存中分配特定的存储空间,以便于用这个空间来存储相关的数据。如果把变量比喻成用于储存数据的盒子,指定数据类型就是指定盒子的大小,既能装得下要装的东西,又不会造成空间的浪费。C语言中变量的数据类型详见表2-2。

表2-2 C语言中的基本数据类型

注:方括号表示其中的内容可以省略。

变量在程序中需要先定义后使用。定义变量的方法是先给变量指定名称和数据类型,这样编译器才能为变量分配相应的存储空间。定义变量的格式如下:


数据类型 变量名表;     //多个变量名称之间要用逗号分隔

在C程序中定义变量的方法可以参考如下语句:


char a,b,c;         //定义a、b、c为三个符号字符型变量
unsigned int num;   //定义num为无符号整型变量

2.1.6 运算符

C语言的运算符非常丰富,在程序中使用这些运算符来处理各种基础操作,从而完成特定的功能。C语言的运算符主要有以下几种。

1.算术运算符

·+:加法运算符。或为取正值运算符。例如:3+5 A+B+23

·–:减法运算符。或为取负值运算符。例如:18–17 TIME1–TIME2–78

·*:乘法运算符。例如:5*8 AD*AF

·/:除法运算符。在这里除法运算符和一般的算术运算规则有所不同,如果是两个浮点数相除,结果也是浮点数。如果两个整数相除,结果也是整数。例如:10.0/20.0结果为0.5,7/2结果为3,而不是3.5。

·%:求余运算符。%两侧均应是整数。例如:10%3结果为1。

在上述的运算符中,我们同样可以用()来改变运算的优先级,这和我们在小学时学的是一样的,如:(A+B)*C,就需要先计算A与B的和,再计算与C的积。

2.赋值运算符

=:赋值运算符。在C语言中用于给变量赋值,其方法可以参考以下语句:


num=25;    //给变量num赋值25
D=C;       //将变量C的值赋给变量D

3.自增、自减运算符

·++:自增运算符。作用是使变量的值自增1。例如:I++,表示让变量I的值自增1。

·––:自减运算符。作用是使变量的值自减1。例如:A––,表示让变量A的值自减1。

4.关系运算符

关系运算符通常是用来判别两个变量是否符合某个条件的,所以使用关系运算符的运算结果只有“真”或“假”,即“1”或“0”两种。

·>:大于。例如:A>B

·<:小于。例如:NUM1

·>=:大于等于。例如:U>=5

·<=:小于等于。例如:P<=7

·==:等于。例如:TEAM1==TEAM2,在这里要区别于赋值运算符“=”,它表示的意思不是将TEAM2的值赋给TEAM1,而是用来判定TEAM1是不是和TEAM2的值相等。

·!=:不等于。例如:A!=B

5.位运算符

位运算是C语言的一大特色。所谓位运算,形象地说就是指把数值以二进制位的方式进行相关的运算,参与位运算的数必须是整型或字符型的数据,实型(浮点型)数不能参与位运算。

1)&:按位与运算符。是实现“必须都有,否则就没有”的运算。它的规则是:

0&0=0 0&1=0 1&0=0 1&1=1

在实际应用中,按位与运算常用来对某些位清0或保留某些位。

例如A的值为:A=1001 0010

只想保留A的高4位,则用:A&1111 0000

与运算后A的值为:1001 0000

2)|:按位或运算符。是实现“只要其中之一有,就有”的运算。它的规则是:

0|0=0 1|0=1 0|1=1 1|1=1

在实际应用中,或运算常用来将一个数值的某些位定值为“1”。

例如A的值为:A=1001 0010

想将A的低4位定值为1,则用:A|0000 1111

或运算后A的值为:1001 1111

3)^:按位异或运算符。是实现“两个不同就有,相同就没有”的运算。它的规则是:

0^0=0 1^0=1 0^1=1 1^1=0

在实际应用中,异或运算常用来使数值的特定位翻转。

例如A的值为:A=1001 1010

想将A的低4位翻转,即0变1,1变0,则用:A^0000 1111

异或运算后A的值为:1001 0101

4)~:按位取反运算符。是实现“是非颠倒”的运算。它的运算规则是:

~0=1 ~1=0

例如A的值为:1001 1010

按位取反运算后,其值为:0110 0101

5)<<:左移运算符。是实现将一个二进制数的每一位都左移若干位的运算。左移运算的方法如图2-2所示。

图2-2 左移运算

6)>>:右移运算符。是实现将一个二进制数的每一位都右移若干位的运算。右移运算的方法如图2-3所示。

图2-3 右移运算

2.1.7 复合赋值运算符

在赋值运算符“=”之前加上其他二目运算符,就可以构成复合赋值运算符。复合赋值运算符有+=、–=、*=、/=、%=、<<=、>>=、&=、^=和|=。

构成复合赋值表达式的方式为:


变量 双目运算符 = 表达式

它相当于:


变量 = 变量 运算符 表达式

例如:

num+=15相当于:num=num+15

a*=b+23相当于:a=a*(b+23)

复合赋值运算符的这种书写方法,对于初学者来说也许不太习惯,但它有利于编译器的编译和处理,可以产生高质量的目标代码。