1、第七章 运算符、表达式、语句我们已经学会如何用变量来表达数据,比如我们现在要写一个工资管理系统我看到很多学员本来昏昏欲睡的眼睛刷地放出了光芒:“老师,我们现在就能写工资管理系统系统了吗?”回答:不能,我们刚刚学了点基础而已。不过,至少我们可以猜想,要写一个工资管理系统,总得懂得如何表达“工资”这个信息吧?还有像职工的年纪,职工人数等等,都得用语言来表达,这些我们现在都会啊!double gongZi; /工资int nianLin; /年龄int zhiGongRenShu; /职工人数看看,上面那些定义变量的代码,你都看得懂,写得出的吧?我们还是颇有成就感的。本章,我们将学习如何对数据进行运
2、算。7.1 算术运算符7.1.1 加减乘除先来学习最基本也最常用的加减乘除等运算。在中,加减乘除分别使用字符 +、-、*、/作为运算符号。加、减、乘的操作没有什么需要特别说明之处,和生活中的相关运算完全一样,如:int a = 1 + 2 - 3 * 4;得到的结果:等于 -9。当然,乘号使用*表示,这你得记。除运算除了使用反余杠/表示以外,很重要的一点是别忘了,对于整数类型,或字符类型的数据进行除运算时,小数部分将被截掉,因为整型类型的数据不能保存小数部分,如:int a = 5 / 2;得到结果:a 等于 2,而不是 2.5。注意:可能大家会以为,之所以 5 /2 结果是 2,是因为我们让
3、一个整型变量 a 等于它,其实原因并不是因为 a 是 int 类型,请看:float a = 5 / 2;虽然 a 现在被声明为实型,但执行这句程序,的值仍然是 2。事实上,精度丢失是在计算机计算 5/2 时就发生了。所以,准确的写法是:float a = 5.0 / 2;或者:float a = 5 / 2.0;或者:float a = 5.0 / 2.0;也就是说,只有除数或被除数至少需要有一个是明确指定为实型,除运算才能得到小数部分。这里我们也更明确类似于5 和 5.0 在计算机中的区别:虽然数值大小一样,但加了 5.0 被当成实型数对待,而 5 则被当成整型数。7.1.2 求模运算除了
4、 + - * / 以外, % 操作也是 C+常用的操作符。% 并不是进行“百分比”的运算。在 C 和 C+里, % 进行求余数运算,求余数也称“求模”,以下是求余操作的例子:int a = 5 % 2;结果是,a 等于 1,即:5 除以,余数为。7.1.3 赋值运算差点忘了,我们已经很熟悉的等号:= ,C,C+称为赋值操作。看看例子,是不是很熟悉:int a = 10;再如:int b;b = a;或:int c = 12 * 2;在中,可以使用连等操作:int a ,b;a = b = 100;结果是,和都为 100。7.1.4 自运算先来看一个计算机编程中常有的语句例子:int a = 1
5、0;a = a + 1;上面的代码执行后,结果的值是 11。可能不是很理解 a = a + 1;这种运算。首先可能会认为, 和 a + 1 怎么会相等呢?这可是个严重错误,要知道,在 C,C+里,= 就是表示赋值操作,至于表示左右两值“相等” 的另有符号。因此,a = a + 1,所完成的工作就是:先计算出 a + 1 的值,然后将该值赋给。假设我们的存款原为,现在存入元,那么新的存款就等于旧存款加上元钱,用编程语言表达,就是 a = a +1;在,中,这样的自加操作可以有另一种表达,并且用这一种表达,计算机的运算速度比较快。a = a +1;的另一种运算速度较快的表达方法:a += 1;+=
6、 被定义为一种新的操作符(因此+和=要连着,中间不能有空格)。它实现的操作就是将其左边的量在自身的基础上加上右边表达式的值。比如:,假设 a 原来的值为 10,那么:a += 2;执行这一句后,a 的值为 12,即 a = 10 + 2;同样的,减,乘,除,求余都有这种操作符:-= 、 *= 、/= 、%= 等。我们以后学习到的另外一些运算符,也有同样的这种对应运算。举一些例子:假设在分别进行以下操作之前,a 原值都为 10。a -= 2; 执行后,值为 8;(a = 10 - 2)a *= 2;执行后,值为 20;(a = 10 * 2) a /= 2;执行后,a 值为 5;(a = 10
7、/ 2)a %= 2;执行后,a 值为 0; (a = 10 % 2),提供这些操作符,目的仅仅是为了提高相应操作的运算速度。为什么 a += 2;会比 a = a + 2;运算得快呢?从编译的角度上看,是因为前者可以生成更短小的汇编代码。,提供这些别的语言没有的操作符,可以供我们写出优化的代码。在某些特殊情况下,优化还可以继续。请看下一小节。7.1.5 + 和 - 运算当运算是自加或自减的时候,提供了更为优化的运算操作符:+,-。设整型变量 a,原值为 10。我们已经知道,要实现对其加,可以有以下两种写法:方法: a = a +1;方法: += 1;我们还知道方法比方法好。现在还有方法,并且
8、是最好的方法。+a,或者:a+;也就是说,在只自加的情况下,代码 a+ 或 +a 可以生成最优化的汇编代码。同样,自减操作也有对应的操作符:-a 或 a-;设 a 原值为 10,则执行 -a 或者 a-后,的值都为。现在来谈谈 +a 和 a+ 有什么区别。在,语言里,+a 和 -a 称为前置运算(prefix),而 a+ 和 a-称为后置运算(postfix)。如果仅仅是进行前置工或后置运算,那么结果是相同的,这我们已经在前面谈过,我们以+为例:设为,则无论是 +a 或 a+,执行结果都是让递增,成为 11。但在有其它运算的复杂表达式中,前置+运算过程是:先加,然后将已加的变量参以其它运算。后
9、置+的运算过程是:先用未加的变量参以其它运算,然后再将该变量加。听起来有些绕,我们举些例子看,还是变量,原来值为 10:例子:int b = +a; /前置运算结果:的值为 11,的值也为 11。计算过程解析:先计算 +a,结果值为 11;然后再计算 b = a;结果值也为 11。例子:int b = a+; /后置运算结果:的值为 11,但的值为 10。计算过程解析:先计算 b = a;因此,b 的值是未加之前的 a,所以为 10;然后再计算 a+,的值为 11。再举一复杂点的表达式:int a = 10;int c = 5;int b = a+ + c;执行这些代码,b 值为 15倘若换成
10、:int a = 10;int c = 5;int b = +a + c;执行这些代码,b 值为 16;想一想,为什么?上在举的是+的例子,对于-操作,其计算顺序的规定一样。+和-的确能加快运算速度,但它们在前置和后置运算上的微小区别,却很空易让你的代码变得不清晰。更为不好的是,不同的编译器可能会对比有不同的解释,比如 VC 和 BC/CB 会对同一代码会有不同的编译结果,造成代码的运行结果也不一样,这是我们应该尽量避免的。所以我们建议在代码尽量不要依赖于前置和后置运算的区别。(尽管它会让你的代码看上去很象“高手”所写)。7.2 算术类型转换7.2.1 隐式类型转换类型转换在 C,C+中也属于
11、一种运算。前面我们举过一个例子:float a = 5 / 2 ;还记得 a 的计算结果吗?上式中,a 将得到的值是 2。因为在除式 5/2 中,5 和 2 都是整数,所以计算机按整数类型进行了除运算,结果所有的小数位都被丢失了。我们列出了三种可以解决小数位丢失的方法:方法 1: float a = 5.0 / 2;方法 2: float a = 5 / 2.0;方法 3: float a = 5.0 / 2.0;最后一种方法好理解,5.0 和 2.0 都明确指定为实型(double),所以计算结果可以保存小数位。而像第一种:被除数 5.0被指定的为实型,但除数 2 仍然为整型,这两种数据类型
12、的精度不一样,这时,计算机将按哪个类型作为标准呢?C+遇到两种不同数据类型的数值进行运算时,会将某个数做适当的类型转换,然后再进行转换。转换总是朝表达能力列强的方向进行,并且转换总是逐个运算符进行的。以下是转换的两条方向线:char/unsigned char short/unsigned short int/unsigned int double long doublefloat double long double像上面的 a = 5 / 2。计算机先计算 5/2,由于 5,2 一样是整型,所以计算机不作转换,算出其结果为 2,然后赋值给a,因此,就算 a 是 float 类型,但仍然只能
13、得到 2 的结果,而不是 2.5。而 a = 5.0 / 2。计算机在计算 5.0 / 2 时,发现 5.0 是实型(带小数点),而 2 是整型,二者不一,所以将 2 先自动转换成 double 数,然后现和 5.0 进行除运算。这个转换过程,在程序运行时自动进行,称为隐式转换。隐式(自动)转换尽量使用我们的程序更加合理,但有时它并不能完全符合我们的要求。比如:int b = 5, c = 2;float a = b / c;由于除号两边的操作数:b、c 都是有明确类型的变量。这时,既不会有隐式转换进行,我们也不能通过加 .0来改变其中某个数的数据类型:float a = b.0 / c.0;
14、 /这种写法是错误的,不可能实现。这种情况下,我们需要显式(强制)类型转换。7.2.2 显式类型转换显式类型转换也称为强制类型转换。它的语法形式有两种:形式 1 : (类型名) 变量或数值形式 2: 类型名(变量或数值)实际例子如:int b = 5, c = 2;float a = (float)b / c;或者:float a = float(b) / c;两种写法都可以将变量 b 强制转换为 float 类型。不过,在要转换的数据类型带有修饰符时,则必须使用第一种型式。比如:(unsigned int) a;其实,两边都加上括号有时更清晰:(unsigned int) (a);7.3 关
15、系运算“关系”运算?听上去很费解。计算机系一师哥师妹正在处朋友,某晚两人在校园林荫处正在谈情说爱,突然冒出一校监:“说!你俩什么关系?”果然不愧为计算机系的一对小情侣,以下是他们的回答:男:“我比她高!”女:“我比他瘦。”男:“我比她壮!”女:“我比他美。”校监:“我倒!”所谓的关系运算,在 C,C+语言里,就是比较运算。算术运算所得的结果是数值,而关系运算所得的结果为逻辑值,也称布尔值。即我们以前所学的 bool 类型允许的值:真或假。真用 true 表示,假用 false 表示。关系操作符有:= (比较左右值是否相等) (比较左值是否大于右值)= (比较左值是否大于或等于右值,也称为不小于
16、),请大家注意。下面举一些例子。int a = 10;int b = 9;则:a = b+1 运算结果: true;a = b 运算结果: false;a b 运算结果: truea = b 运算结果: true;b a 运算结果: false;a = b+1 运算结果: true;a = b+1 运算结果: true;a != b; 运算结果:true;7.4 逻辑运算逻辑运算有三个操作符:! (非,取逻辑反,NOT)b = 帅胜德华留三分;见面条件: a 和比尔盖茨一样富并且和刘德华一样帅,这样的人有吗?不管你嫁得出去退不出去,现在你必须牢牢记住:表达式:条件 1 b = 帅胜德华留三分;
17、见面条件: a | b;事情发生了巨大的变化。现在,有个丑八怪,但却和比尔盖茨一样富,你就得会会他老人家。另外有一穷光蛋,却貌如潘安(假设潘安比德仔帅点),按照条件,你也得去见见。因为你列的条件是:a 或者 b,只要其一成立,总体条件即成立。最后我们来说说“!”。感叹号在 C,C+拿来表达“相反”或“非、不”等意思。它的运行很简单:原来为真,加感叹号后为假,原来为假,加感叹号后为真。下表列出三种逻辑操作符的使用方法:符号 意思 例子 其中,R 是某一定义的变量,表示半径,而 PAI 我们事先定义的一个值为 3.14 的宏。PAI * R * R; 是一个表达式;area = PAI * R *
18、 R;也是一个表达式。表达式组成了 C,C+语句,而语句组成 C,C+的程序。简单的如:3 + 2 ,也是一个表达式。表达式是操作符、操作数和标点符号组成的序列,用于表达一个计算过程。对表达式的计算,需要考虑其各计算部分的运算优先级,其中最熟悉莫过于我们小学就学过的“括号优先,先乘除后加减”。下面列出我们已学过的运算符的优先级:按优先级高低排列的运算符:级别 运算符 说明1 ( ) 括号2 ! +(正号) -(负号) + - sizeof +,-在这里不是加减,而是指正负号3 * / % 乘,除,求模(取余)4 + - 加减5 = != 等于 不等于判断6 计算机可以执行该语句,但它并不改变程
19、序的运行逻辑。就像我们说话时说了一句废话。当一些表达式组合起来,完成某一相对完整的功能后,再加一个分号表示结束,这就组成一条语句。如:a = 3 +2;看,这是一行赋值语句,它改变了 a 的值。当然,语句也可以直接是一个分号,称为空语句:;除非为了调试方例,否则写一句空语句纯属多余。我们已经学习过两种常用的语句类型,其一为变量定义语句,其二为赋值语句。变量定义语句完成指定变量的定义。int a ,b,c;赋值语句则实现为指定变量获得指定值的操作。a = 20;b = 10 * 2 / 3;c = 2 * (a + b);如上面所示,赋值时,右值(等号右边的值),可以是一简单的常数或变量,也可以
20、是一个表达式。在 C,C+中,赋值语句可以使用连等:a = b = 10;执行这一语句时,b 先等于 10,然后 a 等于 b 的值,结果 a 和 b 都是 10。当然,变量定义语句也可以和赋值语结合,即我们以前学的,在定义变量时同时初始化。int a = 10,b = 20;int c = a * b;不过,你不能在定义变量时同时使用连等来实现初始化:int a = b = 10; /错误。编译器会报错说,b 还没定义。有时候,连续的多句语句属于同一控制范围,这时,我们用一对花括号将这些语句括起:int a = 100;int b = a * 20;花括号内的内容,称为复合语句。正是一行行语句组成 C,C+程序,结果本章时,我们可以自豪地宣布:我们已经一脚迈入了 C+大门的门槛!到页首