三、操作符
1.优先级
当一个表达式中存在多个操作符时,操作符的优先级就决定了各部分的计算顺序。程序员常常忘记其他优先级规则,所以应该用括号明确规定计算顺序。
当编译器观察到一个String后面紧跟着一个“+”,此时这个“+”意味着“字符串连接”,并且如果必要,它还要执行“字符串转换”。
2.赋值
赋值采用操作符“=”。它的意思是“取右边的值,把它复制给左边”。右边可以是任何常数、变量或表达式。但左边必须是一个明确的、已命名的变量。
对基本类型的赋值是很简单的。基本类型存储了实际的数值,而并非指向一个对象的引用,所以在为其赋值的时候,是直接将一个地方的内容复制到了另一个地方。但是在为对象“赋值”时,情况却发生了变化。对一个对象进行操作时,我们真正操作的是对对象的引用。所以倘若“将一个对象赋值给另一个对象”,实际是将“引用”从一个地方复制到另一个地方。
注意:别名现象。
3.算数操作符
Java的基本算术操作符与其他大多数程序设计语言是相同的。
4.自动递增和递减
前缀递增和前缀递减(如++a和--a),会先执行运算,再生成值。
后缀递增和后缀递减(如a++和a--),会先生成值,再执行运算。
5.直接常量
直接常量后面的后缀字符标志了它的类型。若为大写(或小写)的L,代表long。大写(或小写)字母F,代表float。大写(或小写)字母D,则代表double。
十六进制数适用于所有整数数据类型,以前缀0x(或0X),后面跟随0-9或小写(或大写)的a-f来表示。如果试图将一个变量初始化成超出自身表示范围的值,编译器都会向我们报告一条错误信息。如果超出范围,编译器会将值自动转换成int型,并告诉我们需要对这次赋值进行“窄化转型”。
八进制数由前缀0以及后续的0-7的数字来表示。
Java采用一种很不直观的记数法来表示指数,e代表“10的幂次”。
在Java中,二进制数没有直接常量表示方法。但是,在使用十六进制和八进制记数法时,以二进制形式显示结果将非常有用。通过使用Integer和Long类的静态方法toBinarySting()可以很容易地实现这一点。
6.移位操作符
移位操作符的运算对象是二进制的“位”。移位操作符只可用来处理整数类型。左位移操作符(<<)能按照操作符右侧指定的位数将操作符左侧的操作数向左移动(在低位补0)。“有符号”右移位操作符(>>)则按照操作符右侧指定的位数将操作符左侧的操作数向右移动。“有符号”右移位操作符使用“符号扩展”:若符号为正,则在高位插入0;若符号为负,则在高位插入1。Java中加入了一种“无符号”右移位操作符(>>>),它使用“零扩展”:无论正负,都在高位插入0。
如果对char、byte或者short类型的数值进行移位处理,那么在移位之前,它们会被转换成int类型,并且得到的结果也是一个int类型的数值。
printBinaryInt(),接受int类型的参数,并用二进制格式输出,同时附有简要的说明文字。
数字的二进制表示形式称为“有符号的二进制补码”。
7.三元操作符 if-else
三元操作符也称为条件操作符,有三个操作数,最终会生成一个值。
其形式为: boolean-exp ? value0 : value1
如果boolean-exp(布尔表达式)的结果为true,计算value0,否则计算value1,计算结果就是这个表达式的值。
8.字符串操作符 + 和 +=
这两个操作符在Java中有一项特殊用途:连接不同的字符串。
Java程序员不能实现自己的运算符重载(如C++和C#那样)。
如果一个表达式以一个字符串起头,那么后续所有操作数都必须是字符串型(编译器会自动把双引号内的字符序列转成字符串)。
9.常犯错误
使用操作符时一个常犯的错误就是,即使对表达式如何计算有点不确定,也不愿意加括号。
Java不会自动地将int数值转换成布尔值。所以 if ( x=10 ); 这种语句是非法的。
10.类型转换操作符
类型转换(cast)的原意是“模型铸造”。在适当的时候,Java会将一种数据类型自动转换成另一种。
要想执行类型转换,需要将希望得到的数据类型置于圆括号内,放在要进行类型转换的值的左边,可以在下面的示例中看到它:long a = (long)250;
即可对数值进行类型转换,亦可对变量进行类型转换。
如果要执行一种名为窄化转换的操作(也就是说,将能容纳更多信息的数据类型转换成无法容纳那么多信息的类型),就有可能面临信息丢失的危险。此时,编译器会强制我们进行类型转换。而对于扩展转换,则不必进行显式地进行类型转换,因为新类型肯定能容纳原来类型的信息,不会造成任何信息的丢失。
布尔型不允许任何的类型转换。“类”数据类型不允许进行类型转换。为了将一种类转换成另一种,必须采用特殊的方法(对象可以在其所属的类型的类族之间可以进行类型转换)。
截尾和舍入
在进行窄化转换时,必须注意截尾和舍入问题。例如:29.7转换成int,是30还是29?
答案是在将float或double转型为整型值时,总是对该数字进行截尾。如果想得到舍入的结果,就需要使用java.lang.Math中的round()方法。
提升
如果对基本数据类型执行算术运算或者按位运算,只要类型比int小,那么在运算之前,这些值会自动转换成int。最终生成的结果就是int类型。
通常,表达式中出现的最大的数据类型决定了表达式最终结果的数据类型。
11.Java没有sizeof
Java不需要sizeof()操作符,因为所有数据类型在所有机器中的大小是相同的。
12.操作符小结
能够对布尔值进行的运算非常有限。我们只能赋予它true和false值,并测试其真假值,而不能相加或是其他任何运算。
在char、byte和short中,对这些类型中的任何一个进行算术运算,都会获得一个int结果,必须将其显式地类型转换回原来的数据类型,以将值赋给原本的类型。
除了boolean外,任何一种基本类型都可通过类型转换变为其他基本类型。当类型转换成一种较小的类型时,必须留意“窄化转换”的结果;否则会在不知不觉中丢失了信息。