zoukankan      html  css  js  c++  java
  • 【面试题】从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)综上分析,结果和预期相同。

  • 相关阅读:
    软件测试流程
    Python2 RF(3.0.4)与Python3 RF(3.1.2)区别
    Ubuntu Install RobotFramework with Python3
    Beta测试与Alpha测试有什么区别
    网络协议,如TCP/UDP的区别?
    缺陷相关知识
    linux_machine-id
    monkey自定义脚本实践
    Monkey事件
    Linux虚拟机fdisk分区
  • 原文地址:https://www.cnblogs.com/SherLocked/p/12795632.html
Copyright © 2011-2022 走看看