运算符与基本运算
2.4.1算术运算符与算术表达式
+(加法运算符,或正值运算符。如3+5,+3)
-(减法运算符,或负值运算符。如5-2,-3)
* (乘法运算符。如3*5)
/(除法运算符。如5/3)
%(模运算符,或称求余运算符,%两侧均应为整型数据,如7%4的值为3)。
自增(++)
++i(在使用i之前,先使i的值加1,如果i的原值为3,则执行j=++i后,j的值为4)
i++ (在使用i之后,使i的值加1,如果i的原值为3,则执行j=i++后,j的值为3,然后i变为4)
自减(--)
--i (在使用i之前,先使i的值减1,如果i的原值为3,则执行j=--i后,j的值为2)
i--(在使用i之后,使i的值减1,如果i的原值为3,则执行j=i--后,j的值为3,然后i变为2)
请注意:
(1) 自增运算符(++)和自减运算符(--)只能用于变量,而不能用于常量或表达式。
(2) ++和--的结合方向是“自右至左”,见附录B。
(3) 自增运算符(++)和自减运算符(--)使用十分灵活,但在很多情况下可能出现歧义性,产生“意想不到”的副作用。
(4) 自增(减)运算符在C++程序中是经常见到的,常用于循环语句中,使循环变量自动加1。也用于指针变量,使指针指向下一个地址。
2.4.2 强制类型转换运算符
在表达式中不同类型的数据会自动地转换类型,以进行运算。有时程序编制者还可以利用强制类型转换运算符将一个表达式转换成所需类型。例如:
(double)a (将a转换成double类型)
(int)(x+y) (将x+y的值转换成整型)
(float)(5%3) (将5%3的值转换成float型)
强制类型转换的一般形式为
(类型名)(表达式)
注意: 如果要进行强制类型转换的对象是一个变量,该变量可以不用括号括起来。如果要进行强制类型转换的对象是一个包含多项的表达式,则表达式应该用括号括起来。如果写成
(int)x+y
则只将x转换成整型,然后与y相加。
以上强制类型转换的形式是原来C语言使用的形式,C++把它保留了下来,以利于兼容。C++还增加了以下形式:
类型名(表达式)
如int(x) 或 int(x+y)
类型名不加括号,而变量或表达式用括号括起来。这种形式类似于函数调用。但许多人仍习惯于用第一种形式,把类型名包在括号内,这样比较清楚。
需要说明的是在强制类型转换时,得到一个所需类型的中间变量,但原来变量的类型未发生变化。例如:
(int)x
如果x原指定为float型,值为3.6,进行强制类型运算后得到一个int型的中间变量,它的值等于3,而x原来的类型和值都不变。
例2.4 强制类型转换。
#include <iostream>
using namespace std;
int main( )
{ float x;
int i;
x=3.6;
i=(int)x;
cout<<″x=″<<x<<″,i=″<< i<<endl;
return 0;
}
运行结果如下:
x=3.6,i=3
x的型仍为float型,值仍等于3.6。
由上可知,有两种类型转换,一种是在运算时不必用户指定,系统自动进行的类型转换,如3+6.5。第二种是强制类型转换。当自动类型转换不能实现目的时,可以用强制类型转换。此外,在函数调用时,有时为了使实参与形参类型一致,可以用强制类型转换运算符得到一个所需类型的参数。
2.4.3赋值运算符与赋值表达式
赋值符号“=”就是赋值运算符,它的作用是将一个数据赋给一个变量。如“a=3”的作用是执行一次赋值操作(或称赋值运算)。把常量3赋给变量a。也可以将一个表达式的值赋给一个变量。
赋值过程中的类型转换
如果赋值运算符两侧的类型不一致,但都是数值型或字符型时,在赋值时会自动进行类型转换。
(1) 将浮点型数据(包括单、双精度)赋给整型变量时,舍弃其小数部分。
(2) 将整型数据赋给浮点型变量时,数值不变,但以指数形式存储到变量中。
(3) 将一个double型数据赋给float变量时,要注意数值范围不能溢出。
(4) 字符型数据赋给整型变量,将字符的ASCII码赋给整型变量。
(5) 将一个int、short或long型数据赋给一个char型变量,只将其低8位原封不动地送到char型变量(发生截断)。例如
short int i=289;
char c;
c=i; //将一个int型数据赋给一个char型变量
赋值情况见图2.8。为方便起见,以一个int型数据占两个字节(16位)的情况来说明。
(6) 将signed(有符号)型数据赋给长度相同的unsigned(无符号)型变量,将存储单元内容原样照搬(连原有的符号位也作为数值一起传送)。
例2.5 将有符号数据传送给无符号变量。
#include <iostream>
using namespace std;
int main( )
{ unsigned short a;
short int b=-1;
a=b;
cout<<″a=″<<a<<endl;
return 0;
}
运行结果为65535
赋给b的值是-1,怎么会得到65535呢?请看图2.9所示的赋值情况。
-1的补码形式为1111111111111111(即全部16个二进制位均为1),将它传送给a,而a是无符号型变量,16个位全1是十进制的65535。如果b为正值,且在0~32767之间,则赋值后数值不变。
不同类型的整型数据间的赋值归根结底就是一条:按存储单元中的存储形式直接传送。
C和C++使用灵活,在不同类型数据之间赋值时,常常会出现意想不到的结果,而编译系统并不提示出错,全靠程序员的经验来找出问题。这就要求编程人员对出现问题的原因有所了解,以便迅速排除故障。
复合的赋值运算符
在赋值符“=”之前加上其他运算符,可以构成复合的运算符。如果在“=”前加一个“+”运算符就成了复合运算符“+=”。例如,可以有
a+=3 等价于 a=a+3
x*=y+8 等价于 x=x*(y+8)
x%=3 等价于 x=x%3
以“a+=3”为例来说明,它相当于使a进行一次自加3的操作。即先使a加3,再赋给a。同样,“x*=y+8”的作用是使x乘以(y+8),再赋给x。
为便于记忆,可以这样理解:
凡是二元(二目)运算符,都可以与赋值符一起组合成复合赋值符。C++可以使用以下几种复合赋值运算符:
+=,-=,*=,/=,%=,<<=,>>=,&=,∧=,|=
其中后5种是有关位运算的。
C++之所以采用这种复合运算符,一是为了简化程序,使程序精炼,二是为了提高编译效率(这样写法与“逆波兰”式一致,有利于编译,能产生质量较高的目标代码)。专业的程序员在程序中常用复合运算符,初学者可能不习惯,也可以不用或少用。
赋值表达式
由赋值运算符将一个变量和一个表达式连接起来的式子称为“赋值表达式”。
它的一般形式为
<变量> <赋值运算符> <表达式>
如“a=5”是一个赋值表达式。对赋值表达式求解的过程是:先求赋值运算符右侧的“表达式”的值,然后赋给赋值运算符左侧的变量。一个表达式应该有一个值。赋值运算符左侧的标识符称为“左值”(left value,简写为lvalue)。并不是任何对象都可以作为左值的,变量可以作为左值,而表达式a+b就不能作为左值,常变量也不能作为左值,因为常变量不能被赋值。
出现在赋值运算符右侧的表达式称为“右值”(right value,简写为rvalue)。显然左值也可以出现在赋值运算符右侧,因而左值都可以作为右值。如
int a=3,b,c;
b=a; // b是左值
c=b; // b也是右值
赋值表达式中的“表达式”,又可以是一个赋值表达式。如
a=(b=5)
下面是赋值表达式的例子:
a=b=c=5 (赋值表达式值为5,a,b,c值均为5)
a=5+(c=6) (表达式值为11,a值为11,c值为6)
a=(b=4)+(c=6) (表达式值为10,a值为10,b等于4,c等于6)
a=(b=10)/(c=2) (表达式值为5,a等于5,b等于10,c等于2)
请分析下面的赋值表达式:
(a=3*5)=4*3
赋值表达式作为左值时应加括号,如果写成下面这样就会出现语法错误:
a=3*5=4*3
因为3*5不是左值,不能出现在赋值运算符的左侧。
赋值表达式也可以包含复合的赋值运算符。如
a+=a-=a*a
也是一个赋值表达式。如果a的初值为12,此赋值表达式的求解步骤如下:
① 先进行“a-=a*a”的运算,它相当于a=a-a*a=12-144=-132。
② 再进行“a+=-132”的运算,它相当于a=a+(-132)=-132-132=-264。
C++将赋值表达式作为表达式的一种,使赋值操作不仅可以出现在赋值语句中,而且可以以表达式形式出现在其他语句(如输出语句、循环语句等)中。这是C++语言灵活性的一种表现。
请注意,用cout语句输出一个赋值表达式的值时,要将该赋值表达式用括号括起来,如果写成“cout<<a=b;”将会出现编译错误。
2.4.4逗号运算符与逗号表达式
C++提供一种特殊的运算符——逗号运算符。用它将两个表达式连接起来。如
3+5,6+8
称为逗号表达式,又称为“顺序求值运算符”。逗号表达式的一般形式为
表达式1,表达式2
逗号表达式的求解过程是:先求解表达式1,再求解表达式2。整个逗号表达式的值是表达式2的值。如,逗号表达式
a=3*5,a*4
从附录B可知:赋值运算符的优先级别高于逗号运算符, 因此应先求解a=3*5(也就是把“a=3*5”作为一个表达式)。经计算和赋值后得到a的值为15,然后求解a*4,得60。整个逗号表达式的值为60。
一个逗号表达式又可以与另一个表达式组成一个新的逗号表达式,如
(a=3*5,a*4),a+5
逗号表达式的一般形式可以扩展为
表达式1,表达式2,表达式3,…,表达式n
它的值为表达式n的值。
从附录B可知,逗号运算符是所有运算符中级别最低的。因此,下面两个表达式的作用是不同的:
① x=(a=3,6*3)
② x=a=3,6*a
其实,逗号表达式无非是把若干个表达式“串联”起来。在许多情况下,使用逗号表达式的目的只是想分别得到各个表达式的值,而并非一定需要得到和使用整个逗号表达式的值,逗号表达式最常用于循环语句(for语句)中,详见第3章。
在用cout输出一个逗号表达式的值时,要将该逗号表达式用括号括起来,如
cout<<(3*5,43-6*5,67/3)<<endl;
C和C++语言表达能力强,其中一个重要方面就在于它的表达式类型丰富,运算符功能强,因而使用灵活,适应性强。