The following code prints out \"3\", not \"4\" as you might expect.
public class Foo2 {
public static void main(String[] args) {
int a=1, b=2;
what the postfix ++ operator is saying is:
Use the original value of the variable in whatever equation, and then increment the variable afterwards.
Postfix ++
increments the value of variable, and returns the value that was there before the increment. Thus, the return value of operator++
in your example will be 1
, and of course 1 + 2
will give 3
, which is then assigned to a
. By the time of assignment, ++
has already incremented the value of a
to 2
(because of precedence), so =
overwrites that incremented value.
Operator precedence is not being ignored here.
The only slightly confusing thing about a++
is that the postfix ++
operator has two distinct effects:
So if a
has the 1 and b
has the value 2 before this line:
a = b + a++;
Then the following steps happen:
b
b
has the value 2, so remember the value 2a++
a++
has the value 1, so remember the value 1a
by one, so it now holds the value 2a
As you see, the code effectively assigns two values to a
:
a
during the evaluation of a++
a
as a result of the assignmentSince the second assignment happens after the first one, you only see the effect of the second one and you will always observe a
as having the value 3 after that line.
Edit: I'll try to provide an interpretation of the decompiled code. It might be a bit hard to follow, unless you know how the JVM works internally (i.e. you know how that the JVM is a stack-based VM and what that means):
// Push the constant 1 on the stack
0: iconst_1
// Pop the topmost value from the stack (1) and store it in the local variable #1 (a.k.a "a")
1: istore_1
// Push the constant 2 on the stack
2: iconst_2
// Pop the topmost value from the stack (2) and store it in the local variable #2 (a.k.a "b")
3: istore_2
// Load the local variable #2 ("b") and push its value (2) on the stack
4: iload_2
// Load the local variable #1 ("a") and push its value (1) on the stack
5: iload_1
// Increment the local variable #1 by 1 (this action does not use the stack!)
6: iinc 1, 1
// Pop the 2 topmost values from the stack (2 and 1), add them and push the result (3) back on the stack
9: iadd
// Pop the topmost value from the stack (3) and store it in local variable #1 ("a")
10: istore_1
Lines 0-3 simply implement
int a=1, b=2;
The lines 4-10 implement
a = b + a++;
I've left out the other lines, as nothing interesting happens there anymore.
As an interesting sidenote: it's plainly visible that this code is not optimized at all. The reason for this is that optimization is the task of the runtime environment (i.e. the JVM) in the Java world and not of the compiler (javac
for example).
I have never seen
a = b + a++;
being used, it strikes me as bad coding. Using it like that would i think also mean that you could write:
int a++ = 1;
which doesn't work.
normally you would see
int a = 1;
int b = 2;
a = b + a; // 3
a = 1;
a++;
a = b + a; // 4
This isn't a matter of precedence, it's a matter of the definition of the operator. By definition the postfix operator executes after the variable is used in the enclosing expression.
The postincrement/decrement operator (a++) returns the value before the increment. The preincrement/decrement (++a) returns the value after the increment.