JVM方法調用(invokevirtual)


  在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想到的,不知道對不對,如果誰能解開我的這個疑惑真是會十分感謝。

 

  


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM