让我告诉你Java代码中的i++在JVM上是如何执行的

众所周知,JVM运行时数据区域由方法区、虚拟机栈、本地方法栈、堆、程序计数器这些区域构成,那么你真的理解了吗,我这里分享一下i++这个东西在JVM底层实现的的问题,希望对你有帮助。

程序一:

public class Test1 {
    public static void main(String[] args) {
        int i = 8;
        i = i++;
        System.out.println(i);
    }
}

大家想一下这个程序的输出结果是什么,有人说是8,有人说是9,当我们运行结果以后发现结果是8。Why? 为什么?我们先从这段程序的JVM指令上分析一下。

获得这段程序的指令如下所示:

0 bipush 8
 2 istore_1
 3 iload_1
 4 iinc 1 by 1
 7 istore_1
 8 getstatic #2 <java/lang/System.out>
11 iload_1
12 invokevirtual #3 <java/io/PrintStream.println>
15 return

这些指令的相关含义可以从如下地址获取:Chapter 6. The Java Virtual Machine Instruction Seticon-default.png?t=M4ADhttps://docs.oracle.com/javase/specs/jvms/se8/html/jvms-6.html#jvms-6.5.bipush我们来解释下这些指令的含义:

1、bipush 指令将8 压入操作栈中

2、istore_1指令将这个int类型的数据8,从栈中弹出来存储到局部变量表中下标为1的位置上(此时int i = 8代码执行完成,同时局部变量表中有个数值8)。

3、iload_1指令将这个int类型的数据从局部变量表中下标为1的位置上加载出来,放入到操作栈中,此时栈中存在i=8的数据,局部变量表中国也存在i=8的数据

4、iinc 1 by 1 指令将这个局部变量表的i进行加1,此时局部变量表中的i = 9,栈中的i=8

5、关键一步,程序又执行了一次istore_1,则代表指令将操作栈中的数据8,从栈中弹出来存储到局部变量表中下标为1的位置上,此时局部变量表中的9被重写为8

6、最后程序执行加载局部变量表的数据,输出了一个8。

这个过程的图解如下所示:


我相信,当你看完这个图的时候,应该明白这个i++在底层JVM层面的一个执行流程了。

如果你理解了这个图片流程,那么在看下这个程序:

public class Test2 {
    public static void main(String[] args) {
        int i = 8;
        i = ++i;
        System.out.println(i);
    }
}

 大家认为这个程序的结果是什么呢?根据上面的图,在结合该代码的指令,可以知道结果为9。

我们获得的这段程序的指令如下所示:

0 bipush 8
 2 istore_1
 3 iinc 1 by 1
 6 iload_1
 7 istore_1
 8 getstatic #2 <java/lang/System.out>
11 iload_1
12 invokevirtual #3 <java/io/PrintStream.println>
15 return

大家有兴趣的,可以根据这个指令,自己画一画等于9的时候的操作流程,是不是很明白了呢。

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值