今天遇到一个很奇怪的代码,代码如下:
import java.util.*;
public class Test
{
public static void main(String[] args){
int j=0;
int z=10;
for( int i =0; i<100; i++){
j=j++;
}
System.out.println(j);
}
}
我的第一反应这个输出结果应该是100,可执行后发现结果是0。如果将j=j++改为z=j++;那么输出j和z应该是:j=100,z=99,跟我们平时的理解是相同的,那为什么写成j=j++时就结果就是0呢,而相同的代码 在c/c++上无论j=j++,z=j++输出的结果j=100。
下面用javap命令来分析一下简单的例子:
public class Test {
public static void main(String[] args) {
int i = 0;
i = i++;
System.out.println(i);
}
}
通过运行命令:javap –c Test 得到如下结果:
下面简单解释一下:
Code开始
0: iconst_0 //将int类型常量0压入栈(栈顶)
1:istore_1 //将int类型值存入局部变量1
2: iload_1//从局部变量1中装载int类型值 也就是说,这个时候把i变量中的值拿出来了,压入栈(栈顶),因为这里执行的是i++, 也就是先使用后++,所有这句的目的是想把i在++之前的值先找个地方保存起来。
3:iinc 1,1// 把常量值1加到局部变量1上 这句话执行后i变量中存放的值就不是0了,而是1,注意:刚才压入栈的那个值,也就是i在++之前的值,即栈顶的值仍然是0
6:istore_1//把栈顶的int类型的值存入变量1中 这句话就是用栈顶的0把i变量中的1给替换掉了。
所以最后得出了i=0的结果。
用几句通俗易懂的大白话讲一遍过程就是: Java在执行 变量=i++; 在执行变量赋值语句(不是常量赋值,而是例如:i = a;即赋值号后面有变量)之前,会先把i的值找个地方保存起来,然后直接在i变量的空间内对值+1,然后再把刚才保存起来的那个值赋给变量。
再来看看j=i++的情况:
过程是这样的:
0:把int类型的常量0压入栈顶
1:把栈顶int类型的值存入变量1中(即给i赋值为0)
2:把int类型的常量0压入栈顶
3:把栈顶int类型的值存入变量2中(即给j赋值为0)
4:把变量1中的int类型的值装载到栈
5: 把数值1加到变量1中(给i变量的值加1)
8:把栈顶int类型的值存入变量2(即把0赋给j)
所以,这时i的值为1,j的值为0
所以大家在使用java中的i++时一定要注意,其他的没什么问题,只有i=i++;有问题,这和C/C++截然不同。