Operator Precedence in Java(Java运算符优先级)

Java has well-defined rules for specifying the order in which the operators in an expression are evaluated when the expression has several operators. For example, multiplication and division have higher precedence than addition and subtraction. Precedence rules can be overridden by explicit parentheses.

 

Precedence order.

 When two operators share an operand the operator with the higher precedence goes first. For example, 1 + 2 * 3 is treated as 1 + (2 * 3), whereas 1 * 2 + 3 is treated as (1 * 2) + 3 since multiplication has a higher precedence than addition.

 

Associativity.

 When an expression has two operators with the same precedence, the expression is evaluated according to its associativity. For example x = y = z = 17 is treated as x = (y = (z = 17)), leaving all three variables with the value 17, since the = operator has right-to-left associativity (and an assignment statement evaluates to the value on the right hand side). On the other hand, 72 / 2 / 3 is treated as (72 / 2) / 3 since the / operator has left-to-right associativity. Some operators are not associative: for example, the expressions (x <= y <= z) and x++-- are invalid.

 

Precedence and associativity of Java operators.

 The table below shows all Java operators from highest to lowest precedence, along with their associativity. Most programmers do not memorize them all, and even those that do still use parentheses for clarity.

 

LevelOperatorDescriptionAssociativity
16 []
.
()
access array element
access object member
parentheses
left to right
15 ++
--
unary post-increment
unary post-decrement
not associative
14 ++
--
+
-
!
~
unary pre-increment
unary pre-decrement
unary plus
unary minus
unary logical NOT
unary bitwise NOT
right to left
13 ()
new
cast
object creation
right to left
12 * / % multiplicative left to right
11 + -
+
additive
string concatenation
left to right
10 << >>
>>>
shift left to right
9 < <=
> >=
instance of
relational not associative
8 ==
!=
equality left to right
7 & bitwise AND left to right
6 ^ bitwise XOR left to right
5 | bitwise OR left to right
4 && logical AND left to right
3 || logical OR left to right
2 ?: ternary right to left
1  =   +=   -=
*=   /=   %=
&=   ^=   |=
<<=  >>= >>>=
assignment right to left

There is no explicit operator precedence table in the Java Language Specification. Different tables on the web and in textbooks disagree in some minor ways.

 

Order of evaluation of subexpressions.

 Associativity and precedence determine in which order Java applies operators to subexpressions but they do not determine in which order the subexpressions are evaluated. In Java, subexpressions are evaluated from left to right (when there is a choice). So, for example in the expression A() + B() * C(D(), E()), the subexpressions are evaluated in the order A()B()D()E(), and C(). Although C() appears to the left of both D() and E(), we need the results of both D() and E() to evaluate C(). It is considered a poor style to write code that relies upon this behavior (and different programming languages may use different rules).

Short-circuiting. When using conditional and and or operators (&& and ||), Java does not evaluate the second operand unless it is necessary to resolve the result. This allows statements like if (s != null && s.length() < 10) to work reliably. Programmers rarely use the non-short-circuiting versions (& and |) with boolean expressions.

 

The precedence order went awry.

 Sometimes the precedence order defined in a language does not conform with mathematical norms. For example, in Microsoft Excel, -a^b is interpreted as (-a)^b instead of -(a^b). So -1^2 is equal to 1 instead of -1, which is the values most mathematicians would expect. Microsoft acknowledges this quirk as a “design choice.” One wonders whether the programmer was relying on the C precedence order in which unary operators have higher precedence than binary operators. This rule agrees with mathematical conventions for all C operators but fails with the addition of the exponentiation operator. Once the order was established in Microsoft Excel 2.0, it could not easily be changed without breaking backward compatibility.

Exercises.

  1. What is the result of the following code fragment?
    int x = 5;
    int y = 10;
    int z = ++x * y--;
    
  2. What is the result of the following code fragment? Explain.
    System.out.println("1 + 2 = " + 1 + 2);
    System.out.println("1 + 2 = " + (1 + 2));
    
    Answer1 + 2 = 12 and 1 + 2 = 3, respectively. If either (or both) of the operands of the + operator is a string, the other is automatically cast to a string. String concatenation and addition have the same precedence. Since they are left-associative, the operators are evaluated left-to-right. The parentheses in the second statement ensure that the second + operator performs addition instead of string concatenation.
  3. Add parentheses to the following expression to make the order of evaluation more clear.
    year % 4 == 0 && year % 100 != 0 || year % 400 == 0
    
    AnswerLeapYear.java shows a variety of equivalent expressions, including the following reasonable alternative.
    ((year % 4 == 0) && (year % 100 != 0)) || (year % 400 == 0)
    
  4. What does the following code fragment print?
    System.out.println(1 + 2 + "abc");
    System.out.println("abc" + 1 + 2);
    

    Answer3abc and abc12, respectively. The + operator is left-associative, whether it is string concatenation or addition.

posted @   小树木  阅读(372)  评论(0编辑  收藏  举报
编辑推荐:
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
阅读排行:
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· Docker 太简单,K8s 太复杂?w7panel 让容器管理更轻松!

喜欢请打赏

扫描二维码打赏

了解更多

点击右上角即可分享
微信分享提示