关于 i = i++ 两次赋值问题理解
文章目录
前言
以前曾经被一个看似简单,但是答案却很神奇的题目困扰着,题目如下:
public class Test{
public static void main(String[] args){
int i=0;
i=i++;
System.out.println(i);
}
}
于是网上搜了一些文章来解惑,总结了下面3点心得:
1. 区分编译器 操作数栈 和 局部变量值 之间的区别
在赋值运算符 = 执行的时候取出的之前压入操作数栈中的值进行运算,而不是取出局部变量中的值进行运算。i++ 运算会直接将结果先进行运算然后赋值给局部变量,但是运算顺序比 = 早,也就是说 i++ 赋值运算是比较特殊的。
2. 题目本身的特殊性,会对i进行 多次 赋值
i++ 先赋值为了 1 ,但是最后 = 运算符执行之后 i 的值又从操作数栈中取出值重新计算了一次,变为了之前的 0 。
3. 字节码中执行过程如下
public static void main(java.lang.String[]);
Code:
0: iconst_0 //0放到栈顶
1: istore_1 //把栈顶的值保存到局部变量1,也就是i中
2: iload_1 //把i的值放到栈顶,也就是说此时栈顶的值是0
3: iinc 1, 1 //注意这个指令,把局部变量1,也就是i,增加1,这个指令不会导致栈的变化,也就是说局部变量1,即i此时为1了。
6: istore_1 //把栈顶的值(0)保存到局部变量1,也就是让i为0了,所以最后i为0
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
这里举的例子不多,可以参考原文,讲的也是比较清楚,我这里只不过是把关键点列出来了