2014 December 02

Java中间变量缓存陷阱

面试陷阱题:i=i++;

i=i++
system.out.println(i);

先来查看JavaP源码:

Compiled from "Test.java"
class Test {
  Test();
    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_0
       1: istore_1
       2: iload_1                           //此处将i(0)读取到栈中
       3: iinc          1, 1                 //自增      注意自增后并未入栈
       6: istore_1                          //出栈赋值,栈顶值为0
       7: getstatic     #2                  // Field java/lang/System.out:Ljava/
io/PrintStream;
      10: iload_1
      11: invokevirtual #3                  // Method java/io/PrintStream.printl
n:(I)V
      14: return
}

i最终为0; 根据一位大牛的回复:

在这里jvm里面有两个存储区,一个是暂存区(是一个堆栈,以下称为堆栈),另一个是变量区。 语句istore_1是将堆栈中的值弹出存入相应的变量区(赋值);语句iload_1是将变量区中的值暂存如堆栈中。 因为i = i++;是先将i的值(0)存入堆栈,然后对变量区中的i自加1,这时i的值的确是1,但是随后的istore_1又将堆栈的值(0)弹出赋给变量区的i,所以最后i = 0。 又因为i = ++i;是先对变量区中的i自加1,然后再将变量区中i的值(1)存入堆栈,虽然最后执行了istore_1,但也只是将堆栈中的值(1)弹出赋给变量区的i,所以i = ++i;的结果是i = 1。 另外:java i++ ++i 中间变量缓存;访问来源1

故而,最佳实践一:不要在单个的表达式中对相同的变量赋值超过一次。

上一篇
下一篇
Loading Disqus comments...
Table of Contents