前言 說真的,平常看源碼都是自己看完自己懂,很少有寫出來的衝動。 但是在寫演算法的時候,經常用到java中各種集合,其中也比較常用到remove方法。 remove有重載函數,分別傳入參數是索引index或者數據Object(指定泛型後自動轉換),如果指定泛型是其他數據類型還好,但是指定的是Integ ...
前言
說真的,平常看源碼都是自己看完自己懂,很少有寫出來的衝動。
但是在寫演算法的時候,經常用到java中各種集合,其中也比較常用到remove方法。
remove有重載函數,分別傳入參數是索引index或者數據Object(指定泛型後自動轉換),如果指定泛型是其他數據類型還好,但是指定的是Integer或者是int的話,或者就有點懵了。
這曾經也困惑過我,所以我就唯有用實踐解惑了。
測試類設計
- 測試類一
public class Text {
public void remove(int index){
System.out.println("調用傳參為int的remove方法");
}
public void remove(Integer object){
System.out.println("調用傳參為Integer的remove方法");
}
public void remove(Object object){
System.out.println("調用傳參為Object的remove方法");
}
}
- 測試類二
public class Text {
public void remove(Integer object){
System.out.println("調用傳參為Integer的remove方法");
}
public void remove(Object object){
System.out.println("調用傳參為Object的remove方法");
}
}
- 測試類三
public class Text {
public void remove(Object object){
System.out.println("調用傳參為Object的remove方法");
}
}
結果
三個測試類分別傳入int,Integer,Object型變數,觀察效果。
測試類一
傳入類型為int:調用傳參為int的remove方法
傳入類型為Integer:調用傳參為Integer的remove方法
傳入類型為Object:調用傳參為Object的remove方法測試類二
傳入類型為int:調用傳參為Integer的remove方法
傳入類型為Integer:調用傳參為Integer的remove方法
傳入類型為Object:調用傳參為Object的remove方法測試類三
傳入類型為int:調用傳參為Object的remove方法
傳入類型為Integer:調用傳參為Object的remove方法
傳入類型為Object:調用傳參為Object的remove方法
從輸出結果可以看出,當方法的傳參的類層級逐漸變高時,層級較低的傳參會進行向上轉型適應傳參的需要。
原因分析
下麵我們先反編譯各測試類的源碼,結果如下
測試類一
invokevirtual #11 // Method remove:(I)V
invokevirtual #15 // Method remove:(Ljava/lang/Integer;)V
invokevirtual #18 // Method remove:(Ljava/lang/Object;)V
測試類二
invokevirtual #11 // Method remove:(Ljava/lang/Integer;)V
invokevirtual #11 // Method remove:(Ljava/lang/Integer;)V
invokevirtual #17 // Method remove:(Ljava/lang/Object;)V
測試類三
invokevirtual #10 // Method remove:(Ljava/lang/Object;)V
invokevirtual #10 // Method remove:(Ljava/lang/Object;)V
invokevirtual #10 // Method remove:(Ljava/lang/Object;)V
可以看出,反編譯代碼中都是調用實例方法的命令,所以結果中自動"向上轉型"其實是jvm的功勞。jvm通過在編譯時確定調用的傳參類型,靜態分派到具體方法的。
所以在前言中的困惑已經解除了,就是由於jvm中靜態分派的實現,調用次序是int->Integer->Object。
後記
也沒什麼想說的,感覺在閱讀源碼的時候必須多想想為什麼這樣做,為什麼要這樣實現,同時通過斷點或者反編譯的手段找出自己的答案。keep going!
本文首發於cartoon的博客
轉載請註明出處:https://cartoonyu.github.io/cartoon-blog/post/java/java的list介面的remove重載方法調用原理/