舉個例子:
1 public class ExceptionTest{ 2 3 void cantBeZero(int i) throws Exception{ 4 throw new Exception(); 5 6 } 7 8 }
上面代碼編譯后的字節碼指令如下:
1 void cantBeZero(int) throws java.lang.Exception; 2 descriptor: (I)V 3 flags: 4 Code: 5 stack=2, locals=2, args_size=2 6 0: iload_1 7 1: ifne 12 8 4: new #2 // class java/lang/Exception 9 7: dup 10 8: invokespecial #3 // Method java/lang/Exception."<init>":()V 11 11: athrow 12 12: return
1) 其中new指令在java堆上為Exception對象分配內存空間,並將地址壓入操作數棧頂;
2) 然后dup指令為復制操作數棧頂值,並將其壓入棧頂,也就是說此時操作數棧上有連續相同的兩個對象地址;
3) invokespecial指令調用實例初始化方法<init>:()V,注意這個方法是一個實例方法,所以需要從操作數棧頂彈出一個this引用,也就是說這一步會彈出一個之前入棧的對象地址;
4) athrow指令從操作數棧頂取出一個引用類型的值,並拋出;
5) 最后由return指令結束方法。
從上面的五個步驟中可以看出,需要從棧頂彈出兩個實例對象的引用,這就是為什么會在new指令下面有一個dup指令,其實對於每一個new指令來說一般編譯器都會在其下面生成
一個dup指令,這是因為實例的初始化方法肯定需要用到一次,然后第二個留給程序員使用,例如給變量賦值,拋出異常等,如果我們不用,那編譯器也會生成dup指令,在初始化方法調用完成后再從棧頂pop出來。例如我們僅僅創建一個對象而不做任何操作,例如:
1 void cantBeZero(int i) throws Exception{ 2 new Exception(); 3 4 }
上面的代碼僅僅創建了一個Exception對象,而沒有做任何操作。
其編譯后的字節碼指令如下:
1 void cantBeZero(int) throws java.lang.Exception; 2 descriptor: (I)V 3 flags: 4 Code: 5 stack=2, locals=2, args_size=2 6 0: new #2 // class java/lang/Exception 7 3: dup 8 4: invokespecial #3 // Method java/lang/Exception."<init>":()V 9 7: pop 10 8: return
也會生成一個dup指令,只不過在調用完實例初始化方法后,將重復的實例引用又pop出棧了。不過這種情況基本不會出現在我們的代碼中,因為我們創建的每一個對象都應該是有用的。
通過上面的例子你應該比較清楚的理解了為什么創建對象時總會有一個dup指令了。
