public class Main {
public static void main(String[] args) {
int i = 1;
i = i++;
int j = i++;
int k = i + ++i*i++;
System.out.println("i="+i);
System.out.println("j="+j);
System.out.println("k="+k);
}
}
实例代码如上,可以很容易得出答案是i=4,j=1,k=11。
要深究其原理,首先要反编译它的字节码文件,这里我就截取我们要进行论述的部分:
L0/1/2/3 | 分别代表的分别是main方法当中的第1/2/3/4行代码 |
LINENUMBER x | 这里的x表示数字,代表这一段操作码对应的是原代码当中的第x行 |
ICONST_1 | 将int类型的值1入操作数栈 |
ISTORE 1 | 将操作数栈顶的值保存到局部变量1当中 |
ILOAD 1 | 将局部变量1的值装载成int类型入操作数栈 |
IINC 1 1 | 将后一位数的值加到前一位数的局部变量当中去(这里就是将1加到局部变量1当中去) |
IMUL | 将操作数栈顶的两个int类型数相乘,然后结果入栈 |
IADD | 将操作数栈顶的两个int类型数相加,然后结果入栈 |
(1)拿main方法内部第1行代码:int i = 1;来说,它对应的是L0块里的指令操作码:
L0
LINENUMBER 5 L0 //代表原代码当中第5行代码
ICONST_1 //将int类型值为1的数入操作数栈
ISTORE 1 //将操作数栈顶的值保存到变量1当中
对应下图所示过程:
(2)i = i++;
L1
LINENUMBER 6 L1
ILOAD 1 //将局部变量表当中的第一个变量(i)的值入操作数栈
IINC 1 1 //将第一个局部变量也就是i=1加上一个int类型的1,也就变成了i=2
ISTORE 1 //将操作数栈顶值保存为第一个变量(i)
这里i的值曾经变为2,但是后来被覆盖了。
(3)int j = i++;
L2
LINENUMBER 7 L2
ILOAD 1 //将局部变量1入操作数栈
IINC 1 1 //将局部变量1的值加1
ISTORE 2 //将操作数栈栈顶当前的值1存储为局部变量2
(4)int k = i + ++i*i++;
L3
LINENUMBER 8 L3
ILOAD 1 //将局部变量1入操作数栈
IINC 1 1 //将局部变量1的值加1 (此时i=3)
ILOAD 1 //将局部变量1入操作数栈
ILOAD 1 //将局部变量1入操作数栈
IINC 1 1 //将局部变量1的值加1 (此时i=4)
IMUL //将栈顶两个int类型数相乘,结果入栈
IADD //将栈顶两个int类型数相加,结果入栈
ISTORE 3 //将当前栈顶值保存为局部变量3,也就是k
(5)综上分析,结果和预期相同。