一直在学习Java,碰到了不少问题,碰到了不少关于i++和++i的难题,以及最经典的String str = "abc" 共建立了几个对象的疑难杂症。 知道有一日知道了java的反汇编 命令 javap。现将学习记录作一小结,以供本身之后翻看。若是有错误的地方,请指正。java
明确一个问题:javap是什么?网上有人称之为 反汇编器,能够查看java编译器为咱们生成的字节码。经过它,咱们能够对照源代码和字节码,从而了解不少编译器内部的工做。程序员
public class TestJavap { public static void main(String[] args) { int i = 2; int j = 3; } }
这个例子中,咱们只是简单的声明了两个int型变量并赋上初值。下面咱们看看javap给咱们带来了什么:(固然执行javap命令前,你得首先配置好本身的环境,能用javac编译经过了,即:javac -c TestJavap.java )数据结构
咱们只看(方便起见,将注释写到每句后面)学习
Code:
0: iconst_2 //把2放到栈顶
1: istore_1 //把栈顶的值放到局部变量1中,即i中
2: iconst_3 //把3放到栈顶
3: istore_2 //把栈顶的值放到局部变量1中,即j中
4: returnspa
是否是很简单?(固然,估计须要点数据结构的知识) ,那咱们就补点java的关于堆栈的知识:.net
对于 int i = 2;首先它会在栈中建立一个变量为i的引用,而后查找有没有字面值为2的地址,没找到,就开辟一个存放2这个字面值的地址,而后将i指向2的地址。3d
看了这段话,再比较下上面的注释,是否是彻底吻合?code
为了验证上面这一说法,咱们继续实验:对象
public class TestJavap { public static void main(String[] args) { int i = 2; int j = 2; } }
咱们将 i 和 j的值都设为2。按照以上理论,在声明j的时候,会去栈中招有没有字面值为2的地址,因为在栈中已经有2这个字面值,便将j直接指向2的地址。这样,就出现了i与j同时均指向2的状况。blog
拿出javap -c进行反编译:结果以下:
Code:
0: iconst_2 //把2放到栈顶
1: istore_1 //把栈顶的值放到局部变量1中,即i中
2: iconst_2 //把2放到栈顶
3: istore_2 //把栈顶的值放到局部变量2中,即j中(i 和 j同时指向2)
4: return
虽然这里说i和j同时指向2,但这里不等于说i和j指向同一块地址(java是不容许程序员直接修改堆栈中的数据的,因此就不要想着,我是否是能够修改栈中的2,那样岂不是i和j的值都会变化。另:在编译器内部,遇到j=2;时,它就会从新搜索栈中是否有2的字面值,若是没有,从新开辟地址存放2的值;若是已经有了,则直接将j指向这个地址。所以,就算j另被赋值为其余值,如j=4,j值的改变不会影响到i的值。)
public class TestJavap { public static void main(String[] args) { int i = 2; int j = i; } }
仍是javap -c
Code:
0: iconst_2 //把2放到栈顶
1: istore_1 //把栈顶的值放到局部变量1中,即i中
2: iload_1 //把i的值放到栈顶,也就是说此时栈顶的值是2
3: istore_2 //把栈顶的值放到局部变量2中,即j中
4: return
看到这里是否是有点明确了?
既然咱们对javap有了必定的了解,那咱们就开始用它来解决一些实际的问题:
public static void main(String[] args) { int i = 1; i++; int j = 1; ++j; }
反编译结果为
Code:
0: iconst_1
1: istore_1
2: iinc 1, 1 //这个个指令,把局部变量1,也就是i,增长1,这个指令不会致使栈的变化,i此时变成2了
5: iconst_1
6: istore_2
7: iinc 2, 1//这个个指令,把局部变量2,也就是j,增长1,这个指令不会致使栈的变化,j此时变成2了
10: return
能够看出,++在前在后,在这段代码中,没有任何不一样。
咱们再看另外一段代码:
public static void main(String[] args) { int i = 1; i = i++; int j = 1; j = ++j; }
反编译结果:
Code:
0: iconst_1
1: istore_1
2: iload_1
3: iinc 1, 1 //局部变量1(即i)加1变为2,注意这时栈中仍然是1,没有改变
6: istore_1 //把栈顶的值放到局部变量1中,即i这时候由2变成了1
7: iconst_1
8: istore_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
是否看明白了? 若是这个看明白了,那么下面的一个问题应该就是迎刃而解了:
public class TestJavap { public static void main(String[] args) { int m = 0; for (int i = 0; i < 100; i++) { m = m++; } System.out.println(m); } }
m = m ++;这句话,java虚拟机执行时是这样的: m的值加了1,但这是栈中的值仍是0, 立刻栈中的值覆盖了m,即m变成0,所以无论循环多少次,m都等于0。
若是改成m = ++m; 程序运行结果就是100了。。。
share:https://blog.csdn.net/junsure2012/article/details/7099222