大家看一下下面一个程序:

一、问题说明

Test.java

public class Test {
public static void main(String[] args) {
int i = 1;
i = ++i;
System.out.println(i);
}
}

输出:2,这应该没有异议对吧,那么再看一个程序

Test1.java

public class Test1 {
public static void main(String[] args) {
int i = 1;
i = i++;
System.out.println(i);
}
}

输出:1。有点意思了吧~

问题解答

其实这是因为java解析器在生成class文件的不一致造成的。

使用javap -verbose Test 查看内容(代码片段):

public static void main(java.lang.String[]);
Code:
Stack=2, Locals=2, Args_size=1
0: iconst_1 //将1推送到栈顶
1: istore_1 //将栈顶的内容赋值给第二个变量i,注意此时栈顶的内容是:1
2: iinc 1, 1 //执行加一操作
5: iload_1 //将变量i的内容推送到栈顶
6: istore_1 //将栈顶的内容保存到变量i
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

使用 javap -verbose Test1查看内容(代码片段):

public static void main(java.lang.String[]);
Code:
Stack=2, Locals=2, Args_size=1
0: iconst_1
1: istore_1
2: iload_1 //将变量i的内容推送到栈顶,不知道为什么这么生成字节码!注意此时栈顶的内容是:1
3: iinc 1, 1 //执行加一操作
6: istore_1 //将栈顶的内容保存到变量i
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

发现少了一个iload_1,意思是将变量推送至栈顶,结果i=i++;命令虽然执行了加i+1的操作但是没有保存回到变量i中。

命令说明:

  • iconst_1   将int型1推送至栈顶
  • istore_1   将栈顶int型数值存入第二个本地变量
  • iload_1    将第二个int型本地变量推送至栈顶(正是因为少了这个操作,才出现以上问题)
  • iinc     将指定int型变量增加指定值(i++, i--, i+=2)
05-06 17:18