先看一段代码,再解释原因:
public class Demo {
public static void main(String[] args){
int i = 0;
int y=i++;
System.out.println(i);
System.out.println(y);
}
}
打印结果:
i:1
y:0
解释这个问题得从两个地方分析“class指令码”和"jvm的线程栈执行"说明:
1.首先我们查看一下.class反汇编后的指令码:
指令码解释请查询:https://cloud.tencent.com/developer/article/1333540
jvm线程栈执行图(需要对上jvm有一定的了解):
步骤解释:
步骤1:将局部变量表位置为"1(变量i)“的值加载到"操作数栈”
步骤2(关键):iinc指令码并不是直接计算的"操作数栈"中的值,而是直接将局部变量表"位置为1(变量i)"中的值加了1
步骤3:保存值(操作数栈栈顶的值)到局部变量表位置为2(即为变量y)的位置
看到这大家就应该明白原因了,因为iinc并不是直接从操作数栈中直接读取值操作,而是直接操作的变量表里的值。大多数人可能都以为计算型指令码基本都是从操作数栈读取值给"执行器"计算然后再把结果放回"操作数栈",而iinc显然不是。我在了解完jvm线程栈执行过程后就一直在想这个问题,而后才知道遇到计算型指令时并不都是“执行器”与“操作数栈”配合产生结果。
以上是个人结合对jvm的理解结论,不对的还忘指正。