javap与 i++,++i
javap 与 ++i,i++
javap命令
java命令:执行java源代码;
javac命令:将java源代码编译成.class文件
javap命令:对.class文件进行反编译,用于分解class文件.
既然我们对javap有了一定的了解,那我们就开始用它来解决一些实际的问题:
1.i++和++i的问题
执行 javap -verbose TestJavap.class命令 ,反编译结果为:
Code:
0: iconst_1//第一个局部变量 i
1: istore_1 //存储变量1到局部变量表中
2: iinc1, 1 //将局部变量1,也就是i,增加1,这个指令不会导致栈的变化,i此时变成2了
5: iconst_1//第二个局部变量2
6: istore_2//存储变量2到局部变量表
7: iinc2, 1//将局部变量2,也就是j,增加1,这个指令不会导致栈的变化,j此时变成2了
10: return
可以看出,++在前在后,在这段代码中,没有任何不同。
我们再看另一段代码:
反编译结果:
Code:
0: iconst_1 //对应第一个变量 int i
1: istore_1 //将局部变量1的值从操作数栈存储到局部变量表
2: iload_1 //将局部变量1加载到操作栈
3: iinc 1, 1 //局部变量1(即i)加1变为2,注意这时栈中仍然是1,没有改变
6: istore_1 //把栈顶的值放到局部变量1中,即i这时候由2变成了1
7: iconst_1 //对应第二个变量 j
8: istore_2 //将局部变量2的值从操作数栈存储到局部变量表
9: iinc 2, 1 //局部变量2(即j)加1变为2,注意这时栈中仍然是1,没有改变
12: iload_2 //把局部变量2(即j)的值放到栈顶,此时栈顶的值变为2
13: istore_2 //把栈顶的值放到局部变量2中,即j这时候真正由1变成了2
14: return
是否看明白了? 如果这个看明白了,那么下面的一个问题应该就是迎刃而解了:
2.i++与++i
m = m ++;这句话,java虚拟机执行时是这样的: m的值加了1,但这是栈中的值还是0, 马上栈中的值覆盖了m,即m变成0,因此不管循环多少次,m都等于0。
执行 javap命令:
public TestJavap();
descriptor: ()V
flags: ACC_PUBLIC
Code:
stack=1, locals=1, args_size=1
0: aload_0
1: invokespecial #1 // Method java/lang/Object."<init>
":()V
4: return
LineNumberTable:
line 1: 0
public static void main(java.lang.String[]);
descriptor: ([Ljava/lang/String;)V
flags: ACC_PUBLIC, ACC_STATIC
Code:
stack=2, locals=3, args_size=1
0: iconst_0 //对应变量 m
1: istore_1 //将变量m存储在局部变量表中
2: iconst_0 //对应变量i
3: istore_2 //将变量i存储在局部变量表中
4: iload_2 //将变量i加载到操作栈中
5: bipush 100 //将100加入到操作栈中
7: if_icmpge 21 //将i大于等于100,跳到第21步
10: iload_1 //将变量m加载到操作栈中
11: iinc 1, 1 //将局部变量表中的m加1,即0+1;
14: istore_1 //将栈中的变量值(此时为0) 存储到局部变量表中,所以在此之后局部变量表中的值又变成了0
15: iinc 2, 1 //将将局部变量表中的i加1,即1+1;
18: goto 4 //返回第四步,执行
21: getstatic #2 // Field java/lang/System.out:Ljav
a/io/PrintStream;
24: iload_1 //将局部变量表中的m值加载到操作栈中 ,此时为0
25: invokevirtual #3 // Method java/io/PrintStream.prin
tln:(I)V
28: return
如果改为m = ++m; 程序运行结果就是100了。
/**
* @author zhangdi
*/
public class TestJavap {
public static void main(String[] args) {
mAdd();
System.out.println("------------------");
addM();
}
public static void mAdd() {
int m = 0;
for (int i = 0; i < 10; i++) {
m = m++;
System.out.println("m0:" + m);// m0:0
}
System.out.println("m:" + m);// m:0
}
public static void addM() {
int m = 0;
for (int i = 0; i < 10; i++) {
m = ++m;
System.out.println("m0:" + m);// m0:表现出的值为i+1
}
System.out.println("m:" + m);// m:10
}
}