在接触计算机语言基础的时候大多的学习者都接触过i++.++i这种类型的运算。在老师教学和一些课本中,为了让学习者快速记住这种运算的规律,一般有这样的话‘’i++是先运算后加减,++i是先加减后运算‘’。
依照这个思路我们看一个问题
public class Test{
public static void main(String[] args){
int i=0;
i=i++;
System.out.println(i);
}
}
控制台打印的i是0.
我们分析一下:i的初始值为0 ,i=i++时,i先运算,后加减,那么就是先i=0,然后i加一,i=i+1,i=1。但是i=0。这是为什么了???????
其实在上述的运算思路中,有好多冲突的地方。(1)在语言学习中有一条规则,赋值运算符“=‘’,他的运算规则是先运算=好右面的,然后在将右面的值赋值给左边。这条规则是不变的,一条计算机必须遵循的规则,但是i=i++明显违反了这条规则,这是为什么??(2)按照以上计算逻辑结果应该是1,为什么是0??
这是要考虑‘’i++是先运算后加减,++i是先加减后运算‘’这个规律是不是正确了,这,时我们通过反汇编看看底层是咋样的逻辑顺序。
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
这样我们看出,其实i++并不是‘’i++是先运算后加减,++i是先加减后运算‘’这样的规则。这只是片面的一个规则。
我们依据汇编代码理清计算逻辑,在进行计算时,计算机首先将i值放到栈顶,然后i+1,i=1,最后将栈顶的值赋值给i,i就又变为0了。