问题
先来看下列代码输出会有何不同?
public static void main(String[] args) {
int i = 0;
int j = 0;
// i输出为i++
System.out.println("i=" + i++);
// j输出为++j
System.out.println("j=" + (++j));
}
结果为i=0 ; j=1;大家都知道++都是自增加1,初始化变量都相同的情况下那为何i++和++j输出结果不同呢?下面我们一起揭开庐山真面目:
转换为字节码
为了更好的看到效果,我们新建一个类,两个测试方法,分别为TestI与TestJ。TestI计算i++,TestJ计算++j。代码如下:
public class Test {
public int TestI() {
int i = 0;
return i++;
}
public int TestJ() {
int j = 0;
return ++j;
}
}
打开终端(我的是Mac系统)cd到测试文件目录下,然后输入:javac Test.java
会编译Test.java,生成Test.class文件。
然后我们在终端输入:javap -c Test.class命令,查看Test.class的字节码,如下图
public int TestI();
Code:
0: iconst_0 // 将int型(0)推送至栈顶
1: istore_1 // 将数值从操作数栈存储到局部变量1
2: iload_1 // 把局部变量1压入操作数栈(局部变量1的值为0)
3: iinc 1, 1 // 局部变量1自增指令(0+1)
6: ireturn // 返回操作数栈中的数据
public int TestJ();
Code:
0: iconst_0 // 将int型(0)推送至栈顶
1: istore_1 // 将数值从操作数栈存储到局部变量1
2: iinc 1, 1 // 局部变量1自增指令
5: iload_1 // 将局部变量1压入操作数栈(局部变量1的值为0+1)
6: ireturn // 返回操作数栈中的数据
注意iload_1 和 iinc这两个指令,两个方法是颠倒的顺序。
我们知道,ireturn返回的是操作数栈栈顶的数据。i++是先将数值0加载到操作数栈,而后进行针对局部变量表中的数值0自增。所以返回的是操作数栈中的i=0的数值,所以i++结果是0。
而++j相反,先局部变量0自增,后将自增后的值1加载到操作数栈。而后返回操作数栈中的数据,所以++j结果是1。
(自增、自减操作都是直接修改变量的值,不经过操作数栈)
如有错误请大佬给指出~~~