欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

经常出错的一道题

程序员文章站 2022-03-09 16:12:37
...
public class AddTest {
    public static void main(String[] args) {
        int i = 0;
        i = i++;
        System.out.println("i=" + i);
    }
}

输出结果是多少呢?自己思考后继续往下看效果更好哦

=================分界线=================

正确答案是 0

如果能完全想明白,可以结束往下阅读了。如果有疑惑,请继续读下去

我刚看到这道题,给的答案也是 1,因为我以往的理解是 i++ 是先赋值,然后再进行 ++ 操作,在赋值时,i 确实为 0,但 i 应该完成了自增操作,输出的 i 应该为 1 呀,自己也验证了一下,的确是 0。想不明白那就看看字节码吧,也许从中能找到答案

先通过 javac AddTest.java 生成 AddTest.class 文件,然后通过 javap -v AddTest.class 查看具体的字节码(只截取了部分)

经常出错的一道题

i++字节码

首先说明左边的数字序号表示字节码的偏移量(偏移量和前面的字节码长度有关),即当前字节码所在的位置,很像行号,但不是行号

下面我们用图解的方式分析下主要步骤

经常出错的一道题

i++字节码图解

通过图解我们发现操作数栈中的 0 把局部变量表 1 位置的 1 覆盖了,所以最后输出的结果是 0

指令说明

xconst_n (x为 i 表示整数,l 表示长整数,f 表示浮点数,d 表示双精度浮点数,a 表示对象引用;n 为 0~5):常量入栈指令,代表将 n 压入栈

xload_n(x 取值和 const 一样,n 为 0~3):局部变量压栈指令,将第n个局部变量压入操作数栈

xstore_n(x 取值和 const一样,n 为 0~3):出栈装入局部变量表指令,从操作数栈中弹出,赋值给局部变量 n

iinc arg1,arg2:对给定的局部变量做自增操作。执行过程中不需要修改操作数栈。接收两个操作数,第一个为局部变量表的位置,第二个为累加数

++i 呢?

说完了 i++ 就不能落下它的好兄弟 ++i

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

这个结果我想大家都能答对,输出 1,但是我还是想把 ++i 的字节码打出来,然后和 i++ 的进行对比

经常出错的一道题

++i字节码

字节码图解如下所示

经常出错的一道题

++i字节码图解

i++ 与 ++i 对比

我们把这两个图放在一起对比看下

经常出错的一道题

对比

反思

为何一道简单的 i++ 的输出问题,我都能弄错?追根是因为对字节码底层的不了解,虽然我们通过字节码和画图分析出了原因,但这并不够,最好做到看到代码,对应的字节码就映射到脑中。说实话,对我来说挺难的。需要不断练习。懂了字节码会对 Java 中更多底层的知识不止知其然,更知其所以然。