java $p_javap -c命令详解
一直在學習Java,碰到了很多問題,碰到了很多關于i++和++i的難題,以及最經典的String str = "abc" 共創建了幾個對象的疑難雜癥。 知道有一日知道了java的反匯編 命令? javap?,F將學習記錄做一小結,以供自己以后翻看。如果有錯誤的地方,請指正
1.javap是什么
where options include:
-c Disassemble the code
-classpath Specify where to find user class files
-extdirs Override location of installed extensions
-help Print this usage message
-J Pass directly to the runtime system
-l Print line number and local variable tables
-public Show only public classes and members
-protected Show protected/public classes and members
-package Show package/protected/public classes
and members (default)
-private Show all classes and members
-s Print internal type signatures
-bootclasspath Override location of class files loaded
by the bootstrap class loader
-verbose Print stack size, number of locals and args for met
hods
If verifying, print reasons for failure
以上為百度百科里對它的描述,只是介紹了javap的一些參數和使用方法,而我們要用的就是這一個:-c Disassemble the code。
明確一個問題:javap是什么?網上有人稱之為 反匯編器,可以查看java編譯器為我們生成的字節碼。通過它,我們可以對照源代碼和字節碼,從而了解很多編譯器內部的工作。
2.初步認識javap
從一個最簡單的例子開始:
這個例子中,我們只是簡單的聲明了兩個int型變量并賦上初值。下面我們看看javap給我們帶來了什么:(當然執行javap命令前,你得首先配置好自己的環境,能用javac編譯通過了,即:javac TestJavap.java?)
我們只看(方便起見,將注釋寫到每句后面)
Code:0: iconst_2 //把2放到棧頂
1: istore_1 //把棧頂的值放到局部變量1中,即i中
2: iconst_3 //把3放到棧頂
3: istore_2 //把棧頂的值放到局部變量1中,即j中
4: return
是不是很簡單?(當然,估計需要點數據結構的知識) ,那我們就補點java的關于堆棧的知識:
對于 int i = 2;首先它會在棧中創建一個變量為i的引用,然后查找有沒有字面值為2的地址,沒找到,就開辟一個存放2這個字面值的地址,然后將i指向2的地址。
看了這段話,再比較下上面的注釋,是不是完全吻合?
為了驗證上面這一說法,我們繼續實驗:
我們將 i 和 j的值都設為2。按照以上理論,在聲明j的時候,會去棧中招有沒有字面值為2的地址,由于在棧中已經有2這個字面值,便將j直接指向2的地址。這樣,就出現了i與j同時均指向2的情況。
拿出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的值。)
再來一個例子:
還是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有了一定的了解,那我們就開始用它來解決一些實際的問題:
3.i++和++i的問題
反編譯結果為
Code:0: iconst_11: istore_12: iinc 1, 1 //這個個指令,把局部變量1,也就是i,增加1,這個指令不會導致棧的變化,i此時變成2了
5: iconst_16: istore_27: iinc 2, 1//這個個指令,把局部變量2,也就是j,增加1,這個指令不會導致棧的變化,j此時變成2了
10: return
可以看出,++在前在后,在這段代碼中,沒有任何不同。
我們再看另一段代碼:
反編譯結果:
Code:0: iconst_11: istore_12: iload_13: iinc 1, 1 //局部變量1(即i)加1變為2,注意這時棧中仍然是1,沒有改變
6: istore_1 //把棧頂的值放到局部變量1中,即i這時候由2變成了1
7: iconst_18: istore_29: iinc 2, 1 //局部變量2(即j)加1變為2,注意這時棧中仍然是1,沒有改變
12: iload_2 //把局部變量2(即j)的值放到棧頂,此時棧頂的值變為2
13: istore_2 //把棧頂的值放到局部變量2中,即j這時候真正由1變成了2
14: return
是否看明白了? 如果這個看明白了,那么下面的一個問題應該就是迎刃而解了:
m = m ++;這句話,java虛擬機執行時是這樣的: m的值加了1,但這是棧中的值還是0, 馬上棧中的值覆蓋了m,即m變成0,因此不管循環多少次,m都等于0。
如果改為m = ++m; 程序運行結果就是100了。。。
案例2
public classTestJavap {public static voidmain(String[] args) {int i=0;
i= i++ + ++i;
System.out.println(i);
}
}
$ javac TestJavap.java
$ javap-c TestJavap
Compiled from"TestJavap.java"
public class TestJavap extendsjava.lang.Object{publicTestJavap();
Code:0:? aload_01:? invokespecial? #1; //Method java/lang/Object."":()V
4:? return
public static voidmain(java.lang.String[]);
Code:0:? iconst_01:? istore_12:? iload_13:? iinc? ? 1, 16:? iinc? ? 1, 19:? iload_110:? iadd11:? istore_112:? getstatic? ? ? #2; //Field java/lang/System.out:Ljava/io/PrintStream;
15:? iload_116:? invokevirtual? #3; //Method java/io/PrintStream.println:(I)V
19:? return
解釋如下
int i = 0;
i=i++ + ++i;
Code:
0:?? iconst_0? ? ? ? 將 0 推到堆棧中//對應賦值語句 int i = 0;iconst_0中的0為初始值
1:?? istore_1? ? ? ? 從堆棧中彈出這個值,并將它存儲到局部變量表的索引 1 處 。對應賦值語句,上句是賦值,這句是存儲,索引1處即為i,因為沒有其它變量,所以本例中局部變量的索引不變
2:?? iload_1? ? ? ? ?將局部變量表索引 1 處的值推到堆棧中。將局部變量索引1處的計算結果推入堆棧臨時存儲,局部變量的索引從1開始,依此類推,這句意味著將變量i=0先推入堆棧臨時存儲
3:?? iinc??? 1, 1?????? 局部變量表索引 1 處的變量加 1 。將變量i自加1,則此時變量i為1
6:?? iinc??? 1, 1?????? 局部變量表索引 1 處的變量加 1 。再將變量i自加1,則此時變量i為2
9:?? iload_1?????????? 將局部變量表索引 1 處的值推到堆棧中。將變量i=2推入堆棧臨時存儲
10:? iadd?????????????? 從操作數堆棧中彈出兩個整數并讓它們相加。將得到的整數推回堆棧中//將兩次暫存的整數相加,即0+2=2
11:? istore_1??????? 從堆棧中彈出這個值,并將它存儲到局部變量表的索引 1 處。把上步相加之結果彈出堆棧
12:? getstatic?????? #2; //Field java/lang/System.out:Ljava/io/PrintStream;? 對應System.out
15:? iload_1?? ??? ??? ?加載i//將變量i=2推入堆棧臨時存儲
16:? invokevirtual?? #3; //Method java/io/PrintStream.println:(I)V?? ??? ??? ?對應println
19:? return???????????? 退出方法
案例3
public classTest01 {public static voidmain(String[] args) {int a = 1;int b = 1;int c = a +b;
}
}
E:\namespace\test01\src\com\first\ock>javap -c Test01
警告: 二進制文件Test01包含com.first.ock.Test01
Compiled from"Test01.java"
public classcom.first.ock.Test01 {publiccom.first.ock.Test01();
Code:0: aload_01: invokespecial #1 //Method java/lang/Object."":()V
4: return
public static voidmain(java.lang.String[]);
Code:0: iconst_11: istore_12: iconst_13: istore_24: iload_15: iload_26: iadd7: istore_38: return}
參考文章:
總結
以上是生活随笔為你收集整理的java $p_javap -c命令详解的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: java freemarker 分页_1
- 下一篇: java h5 上拉加载更多_移动端H5