i++
When we first learned I ++ and ++ I, we always kept in mind that I ++ is used first and then added 1, ++ I is added first and then used, and we have been moving bricks like this for a long time (this conclusion is of course not wrong), so am I. Until I saw the following title.
public static void func(a){
int i=0;
for(int j=0; j <100; j++){ i = i++; } Stystem.out.println(i);// Print what??
}
Copy the code
I thought I printed 100, and I was wrong, of course. By the way, there is no article. I guess I’m still a bit of a geek. Look at the bytecode!
0 iconst_0
1 istore_0
2 iconst_0
3 istore_1
4 iload_1
5 bipush 100
7 if_icmpge 21 (+14)
10 iload_0
11 iinc 0.1
14 istore_0
15 iinc 1.1
18 goto 4 (- 14)
21 getstatic #2 <java/lang/System.out>
24 iload_0
25 invokevirtual #3 <java/io/PrintStream.println>
28 return
Copy the code
Where 10 to 14 correspond to the bytecode I = I ++. It is highly recommended to install the Jclasslib plugin for IDEA, which makes it easier to specify the meaning of the mnemonic.
- Iload_0: Pushes the value of the local variable with index 0 of the local variable array of the current stack frame onto the operand stack.
- Iinc 0,1: Increments the local variable with index 0 by 1.
- Istore_0: pops the value from the operand stack (denoted as val) and sets the local variable value with index 0 to val.
So if we look at these three mnemonics we can see that iinc is incremented, but it’s not set to I, so I stays the same, I ++ doesn’t do anything at all. Here’s the GIF:
Well, just like that, I is changed and overwritten by the value on the stack
++i
Now let’s change the problem. Let’s change it to ++ I.
public static void func(a){
int i=0;
for(int j=0; j <100; j++){ i = ++i; } Stystem.out.println(i);// Print what??
}
Copy the code
The corresponding bytecode is as follows.
0 iconst_0
1 istore_0
2 iconst_0
3 istore_1
4 iload_1
5 bipush 100
7 if_icmpge 21 (+14)
10 iinc 0.1
13 iload_0
14 istore_0
15 iinc 1.1
18 goto 4 (- 14)
21 getstatic #2 <java/lang/System.out>
24 iload_0
25 invokevirtual #3 <java/io/PrintStream.println>
Copy the code
It’s still the same 10 to 14 lines,
So I can be changed. Print 100.
Mixed operation
In I ++ and ++ I pen tests often mixed use, a pile of + sign headache. Such as: I = (i++) + (i++) + (+ + I) + (+ + I); . I = I ++ + ++ I; Otherwise the bytecode gets confused. The answer to this one is simple, but we just need to understand it better through its bytecode. The bytecode is as follows.
0 iconst_0
1 istore_0
2 iload_0
3 iinc 0.1
6 iinc 0.1
9 iload_0
10 iadd
11 istore_0
12 getstatic #2 <java/lang/System.out>
15 iload_0
16 invokevirtual #3 <java/io/PrintStream.println>
19 return
Copy the code
We only need to focus on lines 2 through 11, and the corresponding values and stacks for these steps are shown below. This is the purpose of this article. Let’s start with a mnemonic.
iadd Both value1 and value2 must be of type int. The values are popped from the operand stack. The int result is value1 + value2. The result is pushed onto the operand stack. The result is the 32 low-order bits of the true mathematical result in a sufficiently wide two’s complement format, represented as a value of type int. If overflow occurs, then the sign of the result may not be the same as the sign of the mathematical sum of the two values. Despite the fact that overflow may occur, execution of an iadd instruction never throws a run-time exception. Well, iAdd will push the value on the stack after adding, remember!!
Hard mode
Let’s draw it I = (I ++)+(I ++)+(++ I)+(++ I); And see if the officer can draw a process diagram. The bytecode is as follows.
0 iconst_0
1 istore_0
2 iload_0
3 iinc 0.1
6 iload_0
7 iinc 0.1
10 iadd
11 iinc 0.1
14 iload_0
15 iadd
16 iinc 0.1
19 iload_0
20 iadd
21 istore_0
22 getstatic #2 <java/lang/System.out>
25 iload_0
26 invokevirtual #3 <java/io/PrintStream.println>
29 return
Copy the code
We only need to focus on lines 2 to 21.