在java代碼運行期間,方法間的調用可以說是最為頻繁的了,那么這些方法間的調用在底層的虛擬機又做了什么事情呢?現在就讓我們揭開那道神秘的面紗。
JVM調用方法有五條指令,分別是invokestatic,invokespecial,invokevirtual,invokeinterface,invokedynamic。invokestatic用來調用靜態方法;invokespecial用來調用私有方法,父類方法(super.),類構造器方法;invokeinterface調用接口方法;invokedynamic方法動態執行;invokevirtual調用所有虛方法,即除了以上的方法外全用invokevirtual調用。
這篇文章主要是說明invokevirtual方法的調用,以一個例子來說明。
class Father{ public void fMe(){ System.out.println("fMe"); fMe1();//invovespecial調用 System.out.println(this); this.fMe1();//invovespecial調用 } private void fMe1(){ System.out.println("fMe1"); } } class Son extends Father{
public void fMe1(){
System.out.println("sMe1");
} } public class ThisTest{ public static void main(String[] args) { Father test = new Son(); test.fMe();//編譯時指向父類中國的fMe(),在運行時由於是invokevirtual調用,因此test將變成實際類型Son,如果Son中有Fme(),就調用Son自己的,若沒有就調用父類的 } }
父類Father中有一個public方法fMe()和一個私有方法fMe1(),子類中沒有對其方法覆蓋,在測試類ThisTest中 Father test = new Son();,並調用fMe(),再在fMe()中調用自己的私有方法fMe1()。在如上的幾個方法調用中test.fMe()是invokevirtual調用,編譯時指向父類中國的fMe(),在運行時由於是invokevirtual調用,因此test將變成實際類型Son,如果Son中有fMe(),就調用Son自己的,若沒有就調用父類的,此時是調用父類的;在父類中 的fMe1()是invovespecial調用。疑問(也是寫這篇文章的目的):上面的this代表Son,既然是invovespecial調用,那么應該是調用Son的fMe1()才對啊,為什么是調用父類的fMe1()。(雖然感覺好無厘頭,明明是private方法了,肯定只有這樣的調用了)
下面給出我自己的理解,不知道對不對!
上面的用this調用的時候,我覺得在編譯期間,this代表的是Father類,而不是Son類,正因為是這樣,在用invovespecial字節碼調用的時候采用在編譯器就確定好了指向父類fMe1()方法,而不是子類的方法。為了確定我說的,我采用了兩種方式去驗證:1是用MyEclipse的動態提示,2是將fMe1()方法改為public,這樣在字節碼指invokevirtual調用的時候看是不是在運行期間改變this為實際類型Son類,即是不是去調用子類的fMe1()方法。
1:我在MyEclipse中用提示鍵得到如下,可以看出只有父類的兩個方法,並沒有子類Son的方法。
2、在我將fMe1()方法改為public后確實是調用的是子類的方法。
class Father{ public void fMe(){ System.out.println("fMe"); this. fMe1();//invovespecial調用 System.out.println(this); this.fMe1();//invovespecial調用 } public void fMe1(){ System.out.println("fMe1"); } } class Son extends Father{ public void fMe1(){ System.out.println("sMe1"); } } public class ThisTest{ public static void main(String[] args) { Father test = new Son(); test.fMe();//編譯時指向父類中國的fMe(),在運行時由於是invokevirtual調用,因此test將變成實際類型Son,如果Son中有Fme(),就調用Son自己的,若沒有就調用父類的 } }
這是在看invokevirtual想到的,不知道對不對,如果誰能解開我的這個疑惑真是會十分感謝。