【面试题】从JVM的角度去理解i++和++i

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)综上分析,结果和预期相同。

posted @ 2020-04-28 17:26  sherlock_221c  阅读(271)  评论(0编辑  收藏  举报