美文网首页java进阶干货程序员Java 进阶
从字节码角度分析 i++ 和 ++i 实现

从字节码角度分析 i++ 和 ++i 实现

作者: 美团Java | 来源:发表于2018-03-22 15:00 被阅读1347次

    在知识星球中,有小伙伴提问,最近看到个面试题:

    int j=0;
    for(int i=0;i<100;i++)
        j = j++;
    System.out.println(j);
    

    输出结果是0,如果换成j++,那么输出100,这是为什么?

    对于这种问题,其实有点经验的程序员都知道,前置++和后置++的运算区别:
    1、前置++是将自身加1的值赋值给新变量,同时自身也加1
    2、后置++是将自身的值赋给新变量,然后才自身加1

    话虽这么说,那怎么来证明上面两点呢?
    万物皆有始有终,对于代码,只能从字节码的角度看看是不是能挖掘有价值的信息。

    在讲字节码之前,先简单的了解下Java栈,在JVM中有这么一个数据结构叫Java栈,当线程启动的时候,会分配一块内存当做该线程的栈,每个栈由一系列的栈帧组成。每个栈帧对应一个方法,当线程执行方法时,就是栈帧出栈入栈的过程。

    每个栈帧包含三部分数据:本地变量(参数+方法内的变量)、操作数栈和其它数据,本文主要涉及本地变量和操作数栈。

    先看看后置++的实现:

    public static void main(String[] args){
        int i= 0;
        i = i++;
    }
    

    编译之后的字节码如下:

    iconst_0  //把数值0 push到操作数栈
    istore_1 // 把操作数栈写回到本地变量第2个位置
    iload_1 // 把本地变量第2个位置的值push到操作数栈
    iinc 1,1  // 把本地变量表第2个位置加1     
    istore_1 // 把操作数据栈写回本地变量第2个位置
    

    整个过程这样的:

    可以发现变量a在执行iinc 1,1的时候已经变成1了,但是istore_1又把变量a所在位置覆盖成了0,所以执行完i = i++,i还是原来那个值。

    另外,来看下前置++的实现:

    字节码就不再解释了,整个过程实现如下:

    和后置++不同的地方在于,在变量进入操作数栈之前,就先执行了iinc指令,所以进入操作数的值是加1后的值,最后写回的值也是最新值。

    早日加入知识星球,早日升值加薪,跳槽double!!!

    相关文章

      网友评论

      • 诚实少年:小狼,我理解的是不是关键的指令是iload_1的位置和incr指令的前后顺序来决定值的结果的?我发现不管怎么样,istore_1到最后都会把值给写回来,从栈里面,所以从栈里面读到的值就尤为重要了。比如先++,然后加载i到栈里面(iload)才是真正想要的值。
      • 秋林格瓦斯:如果换成j++,那么输出100.狼哥这个地方是不是写的有些问题. 是++j
        美团Java:@秋林格瓦斯 哈哈,没什么问题,咨询我的人故意这么写了
      • wyn_做自己:代码下面这一行:“输出结果是0,如果换成j++,那么输出100,这是为什么?” 里面的j++应该是++j吧
      • 侠岚梦:这种指令,只要记一下就可以了
      • NirvanalI:代码下面一行说明有问题?
      • pany1708:istore_1指令的作用是将操作数栈顶的整型值出栈病放到第一个局部变量槽中。刚看书上是这么讲的。
        美团Java:@pany1708 istore_0是第一个
      • 冉桓彬:狼哥,
        那个顺序是不是被jvm规定的?
        遇到这种问题只能拿规定解决校长了吧?
        美团Java:@冉桓彬 嗯,编译器解释的

      本文标题:从字节码角度分析 i++ 和 ++i 实现

      本文链接:https://www.haomeiwen.com/subject/rukhqftx.html