今天看《Java Puzzlers》,看到如下代码
public static void main(String[] args) {
int j = 0;
for (int i = 0; i < 100; i++)
j = j++;
System.out.println(j);
}
当时在想先取j,在赋值给j,j++,这样下来应该是100吧,但是运行代码之后,尽然发现是0!!真是震惊了啊!
于是我在C里面写了同样的代码
int main(int argc, char *argv[]) {
int i, j = 0;
for(i = 0; i < 100; i++)
j = j++;
printf("%d", j);
return 0;
}
输出是100,我一下子彻底凌乱了啊!!!
于是网上搜索答案,说看java的反编译代码,代码如下
public static void main(java.lang.String[]);
Code:
0: iconst_0
1: istore_1 // j = 0
2: iconst_0
3: istore_2 // i = 0
4: goto 15
7: iload_1 //将j压栈
8: iinc 1, 1 //对局部变量j 加1,注意不是对栈中的j加1
11: istore_1 //弹出栈顶元素给j,结果就是j = 0啊
12: iinc 2, 1
15: iload_2
16: bipush 100
18: if_icmplt 7
21: getstatic #16 // Field java/lang/System.out:Ljava/io/PrintStream;
24: iload_1
25: invokevirtual #22 // Method java/io/PrintStream.println:(I)V
28: return
这段反编译代码中最核心的就是先将j压栈,对局部变量j++,栈顶元素弹栈(这里栈顶是j为0)赋值给j,所以整个循环下来j始终为0啊!!
结论:不要在单个表达式中对同一个变量赋值多次,最终的结果可能跟你预想的不一样!OK!继续看书