有题如下:
public class TestTemp {
public static void main(String[] args) {
int i = 1;
i=i++;
System.out.println(i);
}
}
输出结果为1。对于a=b+c++,一般可以看作
- a=b+c;
- c++
按如上理解的话结果应该应该为2的,但是这里为什么结果为1呢?
首先,我们利用javap命令获取JVM字节码命令如下:
public class com.sail.TestTemp {
public com.sail.TestTemp();
Code:
0: aload_0
1: invokespecial #1 // Method java/lang/Object."<init>":()V
4: return
public static void main(java.lang.String[]);
Code:
0: iconst_1
1: istore_1
2: iload_1
3: iinc 1, 1
6: istore_1
7: getstatic #2 // Field java/lang/System.out:Ljava/io/PrintStream;
10: iload_1
11: invokevirtual #3 // Method java/io/PrintStream.println:(I)V
14: return
}
对需要关注内容添加注释如下:
//int i=1;
0: iconst_1 ///常量1入栈
1: istore_1 //将栈顶int类型值保存到局部变量1中。
//i=i++;
2: iload_1 //从局部变量1中装载int类型值入栈。
3: iinc 1, 1 //将整数值1加到指定的int类型的局部变量1中。
6: istore_1 ////将栈顶int类型值保存到局部变量1中。
前两步执行了int i=1;没什么好讲的。
标号2,3,6执行过程:
- 局部变量i入栈,i为1,栈顶为1.
- 局部变量i自增,此时i值为2.
- 加载栈顶元素到i中,i被重新赋值为1.
我们要知道JVM是基于栈的架构。
《深入分析Java Web技术内幕(修订版)》P186
JVM执行字节码指令是基于栈的架构,也就是所有的操作数必须先入栈,然后根据指令中的操作码选择从栈顶弹出若干元素进行计算后再将结果压入栈中。在JVM中操作数可以存放在每一个栈帧中的一个一个本地变量集中,即在每个方法调用时就会给这个方法分配一个本地变量集,这个本地变量集在编译时就已经确定,所以操作数入栈可以直接是常量入栈或者从本地变量集中取一个变量压入栈中。
还需要知道的一点时,一条赋值语句中,赋值操作最后执行。
本文深入探讨Java中自增运算符的工作原理,通过JVM字节码指令详细解释了i=i++为何输出结果为1,而非直观预期的2。揭示了基于栈的架构下,赋值语句中赋值操作的执行时机。

867

被折叠的 条评论
为什么被折叠?



