美文网首页
从字节码层次分析++j和j++的区别

从字节码层次分析++j和j++的区别

作者: 木讷山人_a986 | 来源:发表于2018-03-16 11:38 被阅读0次

    一、缘起

    最近看到个面试题:

    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个位置
    

    整个过程这样的:


    运行过程

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

    接着,来看下前置++i的实现:


    前置++的实现

    整个过程实现如下:


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

    相关文章

      网友评论

          本文标题:从字节码层次分析++j和j++的区别

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