JVM方法調用(invokevirtual)

  在java代碼運行期間,方法間的調用能夠說是最爲頻繁的了,那麼這些方法間的調用在底層的虛擬機又作了什麼事情呢?如今就讓咱們揭開那道神祕的面紗。java

  JVM調用方法有五條指令,分別是invokestatic,invokespecial,invokevirtual,invokeinterface,invokedynamic。invokestatic用來調用靜態方法;invokespecial用來調用私有方法,父類方法(super.),類構造器方法;invokeinterface調用接口方法;invokedynamic方法動態執行;invokevirtual調用全部虛方法,即除了以上的方法外全用invokevirtual調用。測試

  這篇文章主要是說明invokevirtual方法的調用,以一個例子來講明。this

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方法了,確定只有這樣的調用了)spa

  下面給出我本身的理解,不知道對不對!blog

  上面的用this調用的時候,我以爲在編譯期間,this表明的是Father類,而不是Son類,正由於是這樣,在用invovespecial字節碼調用的時候採用在編譯器就肯定好了指向父類fMe1()方法,而不是子類的方法。爲了肯定我說的,我採用了兩種方式去驗證:1是用MyEclipse的動態提示,2是將fMe1()方法改成public,這樣在字節碼指invokevirtual調用的時候看是否是在運行期間改變this爲實際類型Son類,便是不是去調用子類的fMe1()方法。接口

  1:我在MyEclipse中用提示鍵獲得以下,能夠看出只有父類的兩個方法,並無子類Son的方法。ip

二、在我將fMe1()方法改成public後確實是調用的是子類的方法。ci

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想到的,不知道對不對,若是誰能解開個人這個疑惑真是會十分感謝。編譯器

相關文章
相關標籤/搜索