Java 运算符
我们已经学会了如何声明变量和赋值变量。那么你可能想知道如何对它们进行操作。本小节我们学习的运算符就可以对它们进行运算。
运算符是一些特殊的符号,它们对一个,两个或多个操作数执行特定的运算,然后返回一个结果。这里的操作数指的就是运算符操作的实体。
Java
提供了一组丰富的运算符来操作变量。 我们可以将所有Java
运算符分为以下几类:
- 算术运算符
- 关系运算符
- 位运算符
- 逻辑运算符
- 赋值运算符
- 其他运算符
本小节我们将按照以上分类介绍 Java 中的运算符,并在最后介绍运算符的优先级。
1. 算术运算符
1.1 概述
Java 语言提供了执行加减乘除四则运算的运算符。 算数运算符被用在数学表达式中,可以使用任意嵌套的小括号,其作用与数学中相同。下表列出了算术运算符:
(在例子中,初始化两个整型变量a、b:int a = 2;
int b = 4;
)
运算符 |
描述 |
例子 |
+ |
加法运算符 (也用于字符串连接) |
a + b 等于 6 |
- |
减法运算符 |
a - b 等于 -2 |
* |
乘法运算符 |
a * b 等于 8 |
/ |
除法运算符 |
b / a 等于 2 |
% |
取余运算符 |
b % a 等于 0 |
++ |
自增运算符 |
a ++ 等于 3 |
-- |
自减运算符 |
b – 等于 3 |
1.2 实例
num1=5
num2=2
num1自增后:6
num2自减后:1
另外,整型之间的除法运算是整除,这也就意味着2 / 4
的结果为0
,如果我们想像数学一样得到一个小数,可以使用浮点类型的操作数。例如:
整型2除以整型4的结果为:0
浮点型2除以浮点型4的结果为:0.5
要特别注意,在 Java 语言中,算数运算符不能作用于不同类型的操作数。我们来看一个反例:
public class OperatorDemo1 {
public static void main(String[] args) {
boolean b = true;
int i = 20;
int result = i + b;
}
}
javac OperatorDemo1.java
OperatorDemo1.java:7: 错误: 二元运算符 '+' 的操作数类型错误
int result = i + b;
^
第一个类型: int
第二个类型: boolean
1 个错误
编译器给出明确提示:加法运算符的操作数类型错误。这是因为 Java 是强类型的语言,不同类型的操作数的算数运算是违规的,这个原理同样适用于其他算数运算符,此处不再一一列举。
还有一点,自增自减运算符是有前后之分的,++i
表示先加1再引用i
,i++
表示先引用i
再加1。将在下一小节举例介绍。
2. 关系运算符
2.1 概述
关系运算符又称为比较运算符,比较的结果是一个布尔类型的值(true
或false
)。
Java 语言有几个可用于比较变量的运算符,如下表所示:
(在例子中,初始化两个整型变量a、b:int a = 2;
int b = 4;
)
运算符 |
描述 |
例子 |
== |
检查如果两个操作数的值是否相等,如果相等则条件为真。 |
(a == b) 为假 |
!= |
检查如果两个操作数的值是否相等,如果值不相等则条件为真。 |
(a != b) 为真 |
> |
检查左操作数的值是否大于右操作数的值,如果是那么条件为真。 |
(a > b) 为假 |
< |
检查左操作数的值是否小于右操作数的值,如果是那么条件为真。 |
(a < b)为真 |
>= |
检查左操作数的值是否大于或等于右操作数的值,如果是那么条件为真。 |
(a >= b)为假 |
<= |
检查左操作数的值是否小于或等于右操作数的值,如果是那么条件为真。 |
(a <= b)为真 |
Tips:在比较两个操作数是否相等时,必须使用“==
”而不是“=
”。
3.2 实例
id == i1的结果为:true
id != i1的结果为:false
id > i1的结果为:false
id < i1的结果为:false
id >= i1的结果为:true
id <= i1的结果为:true
Tips:>
、<
、>=
、<=
这几个运算符左右两边的操作数必须是byte
,short
,int
,long
,double
,float
,char
这几种数据类型;而==
和!=
运算符的操作数既可以是基本数据类型,又可以是引用数据类型。
3. 位运算符
Java 语言还提供了对整数类型执行按位和移位操作的运算符,称作位运算符。
假设a = 60,b = 13
;它们的二进制格式表示将如下:
a = 0011 1100
b = 0000 1101
-----------------
a & b = 0000 1100
a | b = 0011 1101
a ^ b = 0011 0001
~a = 1100 0011
下表列出了位运算符的基本运算,假设整数变量 a 的值为 60 和变量 b 的值为 13:
运算符 |
描述 |
例子 |
&(按位与) |
如果相对应位都是1,则结果为1,否则为0 |
(a&b),得到12,即0000 1100 |
|(按位或) |
如果相对应位都是 0,则结果为 0,否则为 1 |
(a | b)得到61,即 0011 1101 |
^(按位异或) |
如果相对应位值相同,则结果为0,否则为1 |
(a ^ b)得到49,即 0011 0001 |
〜(按位取反) |
按位取反运算符翻转操作数的每一位,即0变成1,1变成0。 |
(〜a)得到-61,即1100 0011 |
<< (左位移) |
按位左移运算符。左操作数按位左移右操作数指定的位数。 |
a << 2得到240,即 1111 0000 |
>> (右位移) |
按位右移运算符。左操作数按位右移右操作数指定的位数。 |
a >> 2得到15即 1111 |
>>> (零填充右移) |
按位右移补零操作符。左操作数的值按右操作数指定的位数右移,移动得到的空位以零填充。 |
a>>>2得到15即0000 1111 |
4. 逻辑运算符
4.1 概述
逻辑运算符可以在表达式中生成组合条件,例如在执行特定语句块之前必须满足的两个或多个条件。使用逻辑运算符,可以描述这些组合条件。逻辑运算的返回结果只能为真或假。
(在例子中,初始化两个整型变量a、b:int a = 0;
int b = 1;
)
运算符 |
描述 |
例子 |
&&(逻辑与) |
当且仅当两个操作数都为真,条件才为真。 |
(a && b)为假 |
|| (逻辑或) |
如果任何两个操作数任何一个为真,条件为真。 |
(a || b)为真 |
!(逻辑非) |
用来反转操作数的逻辑状态。如果条件为真,则逻辑非运算符将得到假。 |
!(a && b)为假 |
4.2 短路运算
&&
和||
运算符存在短路行为。短路的意思是:只有在需要的时候才会判断第二个操作数的真假。例如:
程序解析:有两个整型变量a
和b
,值分别为1
和2
。第一个if
语句的条件为逻辑与运算,其第一个操作数a == 2
为假,所以无论第二个操作数是真是假,都不去判断,条件直接被判定为假;第二个if
语句的条件为逻辑或运算, 其第一个操作数a == 1
为真,所以无论第二个操作数是真是假,都不会去判断,条件直接被判定为真。这就是所谓的短路。
5. 赋值运算符
5.1 概述
赋值运算符是为指定变量分配值的符号。下标列出了常用 Java 中常用的赋值运算符:
运算符 |
描述 |
例子 |
= |
简单的赋值运算符。将值从右侧操作数分配给左侧操作数。 |
c = a + b将a + b的值赋给c |
+= |
加和赋值运算符。它将右操作数添加到左操作数,并将结果分配给左操作数。 |
c + = a等于c = c + a |
-= |
减和赋值运算符。它从左侧操作数中减去右侧操作数,并将结果分配给左侧操作数。 |
c -= a等效于c = c – a |
*= |
乘和赋值运算符。它将右操作数与左操作数相乘,并将结果分配给左操作数。 |
c *= a等效于c = c * a |
/ = |
除和赋值运算符。它将左操作数除以右操作数,并将结果分配给左操作数。 |
c /= a等于c = c / a |
5.2 实例
初始值:
num1=20 num2=10 result=0
开始赋值运算:
result += num1 结果为:20
result -= num2 结果为:10
result *= num1 结果为:200
result /= num2 结果为:20
result %= 15 结果为:5
6. 其他运算符
6.1 条件运算符(? :)
条件运算符也称为三元运算符。我们会在条件语句小节中再次对其介绍。
该运算符由三个操作数组成,用于判断布尔表达式。它的目的是确定应将哪个值分配给变量。条件运算符的语法为:
如果布尔表达式为真,就将值1
分配变量,否则将值2
分配给变量。
由于age
变量值为15,小于18,age >= 18
返回结果为假,因此编译执行后,屏幕将打印:
6.2 instanceof 运算符
Tips:了解 instanceof 运算符需要一些面向对象的前置知识。目前你可以选择性学习。
instanceof
运算符将对象与指定类型进行比较,检查对象是否是一个特定类型(类类型或接口类型)。
( Object reference variable ) instanceof (class/interface type)
如果instanceof
左侧的变量所指向的对象,是instanceof
右侧类或接口的一个对象,结果为真,否则结果为假。
由于字符串变量name
是String
类型,所以执行代码,屏幕会打印:
注意,instanceof
运算符不能用于操作基本数据类型,如果将字符串类型name
变量改为一个char
类型的变量,编译代码将会报错:
InstanceOfOperators1.java:4: 错误: 意外的类型
boolean b = name instanceof String;
^
需要: 引用
找到: char
1 个错误
7. 运算符的优先级
当多种运算符在一同使用的时候,会有一个执行先后顺序的问题。
下表中的运算符按优先顺序排序。运算符越靠近表格顶部,其优先级越高。具有较高优先级的运算符将在具有相对较低优先级的运算符之前计算。同一行上的运算符具有相同的优先级。
类别 |
操作符 |
关联性 |
后缀 |
() [] . (点操作符) |
左到右 |
一元 |
+ + - !〜 |
从右到左 |
乘性 |
* /% |
左到右 |
加性 |
+ - |
左到右 |
移位 |
>> >>> << |
左到右 |
关系 |
>> = << = |
左到右 |
相等 |
== != |
左到右 |
按位与 |
& |
左到右 |
按位异或 |
^ |
左到右 |
按位或 |
| |
左到右 |
逻辑与 |
&& |
左到右 |
逻辑或 |
| | |
左到右 |
条件 |
?: |
从右到左 |
赋值 |
= + = - = * = / =%= >> = << =&= ^ = | = |
从右到左 |
逗号 |
, |
左到右 |
当相同优先级的运算符出现在同一表达式中时,如何控制它们计算的先后呢。我们来看一个实例:
在计算result
的语句的右侧,+
、-
两个运算符优先级相同,如果我们不加以控制,将按照从左到右顺序计算,打印结果为result = 2
;但是如果我们想先计算a + b
和c + a
的值,再计算两者之差,我们可以使用括号()
将其顺序进行控制:(a + b) - (c + a)
,再执行代码将打印我们想要的结果:result = -2
。
8. 小结
本小节我们按照分类介绍了Java
的运算符,并且在最后给出了运算符的优先级表格。本节内容可能有些繁杂,但细节不必全部记住,你可以在需要的时候回过头来查查表格。当然,最好的方式还是多写多练,对照实例和表格自己敲敲代码,用心分析代码执行的流程,是增强记忆最好的方式。