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

Posted 无处不在的海贼

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了让我告诉你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 Sethttps://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的时候的操作流程,是不是很明白了呢。

以上是关于让我告诉你Java代码中的i++在JVM上是如何执行的的主要内容,如果未能解决你的问题,请参考以下文章

让我来告诉你:最重要的 JVM 参数总结

JVM的理解

了解 JVM 中的对象开销

Java冠军程序员告诉你如何提升技术

IT界的新方向-大数据?让我来告诉你如何从“零”学起!

双亲委派机制