欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java方法調(diào)用解析靜態(tài)分派動態(tài)分派執(zhí)行過程

 更新時間:2022年06月23日 16:52:24   作者:攻城獅Chova  
這篇文章主要為大家介紹了Java方法調(diào)用解析靜態(tài)分派動態(tài)分派執(zhí)行過程,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

方法調(diào)用

在程序運行時,進(jìn)行方法調(diào)用是最普遍,最頻繁的操作

方法調(diào)用不等于方法執(zhí)行:

  • 方法調(diào)用階段唯一的任務(wù)就是確定被調(diào)用的方法版本,即調(diào)用哪一個方法
  • 不涉及方法內(nèi)部的具體運行過程

Class文件的編譯過程不包括傳統(tǒng)編譯中的連接步驟

Class文件中的一切方法調(diào)用在Class文件里面存儲的都是符號引用,而不是方法在在實際運行時內(nèi)存布局中的入口地址,即之前的直接引用:

  • 這樣使得Java具有更強(qiáng)大的動態(tài)擴(kuò)展能力
  • 同時也使得Java方法調(diào)用過程變得相對復(fù)雜
  • 需要在類加載期間,甚至?xí)竭\行期間才能確定目標(biāo)方法的直接引用

方法解析

所有方法調(diào)用中的目標(biāo)方法在Class文件里都是一個常量池的引用

在類的加載解析階段,會將其中的一部分符號引用轉(zhuǎn)化為直接引用:

方法在程序真正執(zhí)行之前就有一個可確定的調(diào)用版本,并且這個方法的調(diào)用版本在運行期是不可改變的

也就是說,調(diào)用目標(biāo)在程序代碼中完成,編譯器進(jìn)行編譯時就必須確定下來,這也叫做方法解析

Java方法分類

在Java中符合 "編譯期可知,運行期不可變" 的方法有兩大類:

  • 靜態(tài)方法: 與類型直接關(guān)聯(lián)
  • 私有方法: 在外部不可被訪問

這兩種方法各自的特點決定這兩種方法都不可能通過繼承或者別的方式重寫版本,因此適合在類加載階段進(jìn)行解析

非虛方法: 在類加載階段會把符號引用解析為該方法的直接引用

  • 靜態(tài)方法
  • 私有方法
  • 實例構(gòu)造器
  • 父類方法

虛方法: 在類加載階段不會將符號引用解析為該方法的直接引用

除去以上的非虛方法,其它的方法均為虛方法

靜態(tài)分派

public class StaticDispatch {
	static abstract class Human {
	}
	static class Man extends Human {
	}
	static class Woman extends Human {
	}
	public static void sayHello(Human guy) {
		System.out.println("Hello, Guy!");
	}
	public static void sayHello(Man guy) {
		System.out.println("Hello, Gentleman!");
	}
	public static void sayHello(woman guy) {
		System.out.println("Hello, Lady!");
	}
	public static void main(String[] args) {
		Human man = new Man();
		Human women = new Woman();
		sayHello(man);
		sayHello(woman);
	}
}

Human man = new Human();

Human為變量的靜態(tài)類型

Man為變量的實際類型

靜態(tài)類型和實際類型在程序中都會放生變化:

靜態(tài)類型:

  • 靜態(tài)類型的變化僅僅在使用時發(fā)生
  • 變量本身的靜態(tài)類型不會被改變
  • 最終的靜態(tài)類型在編譯器中可知

實際類型:

  • 實際類型變化的結(jié)果在運行期才確定下來
  • 編譯器在編譯期間并不知道一個對象的實際類型是什么
Human human = new Man();
sayHello(man);
sayHello((Man)man);		// 類型轉(zhuǎn)換,靜態(tài)類型變化,轉(zhuǎn)型后的靜態(tài)類型一定是Man
man = new woman();		// 實際類型變化,實際類型是不確定的
sayHello(man);
sayHello((Woman)man);	// 類型轉(zhuǎn)換,靜態(tài)類型變化

編譯器在重載時是通過參數(shù)的靜態(tài)類型而不是實際類型作為判斷依據(jù),靜態(tài)類型在編譯期間可以知道:

編譯階段,Javac編譯器會根據(jù)參數(shù)的靜態(tài)類型決定使用哪個重載版本

靜態(tài)分派:

  • 所有依賴靜態(tài)類型來定位方法的執(zhí)行版本的分派動作
  • 典型應(yīng)用 :方法重載

靜態(tài)分派發(fā)生在編譯階段,因此確定靜態(tài)分派的的動作不是由虛擬機(jī)執(zhí)行的,而是由編譯器完成的

由于字面量沒有顯示靜態(tài)類型,只能通過語言上的規(guī)則去理解和推斷

public class LiteralTest {
	public static void sayHello(char arg) {
		System.out.println("Hello, char!");
	}
	public static void sayHello(int arg) {
		System.out.println("Hello, int!");
	}
	public static void sayHello(long arg) {
		System.out.println("Hello, long!");
	}
	public static void sayHello(Character arg) {
		System.out.println("Hello, Character!");
	}
	public static void main(String[] arg) {
		sayHello('a');
	}
}

編譯器將重載方法從上向下依次注釋,得到不同的輸出

如果編譯器無法確定要自定轉(zhuǎn)型為哪種類型,會提示類型模糊,拒絕編譯

public class LiteralTest {
	public static void sayHello(String arg) {	// 新增重載方法
		System.out.println("Hello, String!");
	}
	public static void sayHello(char arg) {	
		System.out.println("Hello, char!");
	}
	public static void sayHello(int arg) {
		System.out.println("Hello, int!");
	}
	public static void sayHello(long arg) {
		System.out.println("Hello, long!");
	}
	public static void sayHello(Character arg) {
		System.out.println("Hello, Character!");
	}
	public static void main(String[] args) {
		Random r = new Random();
		String s = "abc";
		int i = 0;
		sayHello(r.nextInt() % 2 != 0 ? s : 1 );	// 編譯錯誤
		sayHello(r.nextInt() % 2 != 0 ? 'a' : false);	//編譯錯誤
	}
}

動態(tài)分派

public class DynamicDispatch {
	static abstract class Human {
		protected abstract void sayHello();
	}
	static class Man extends Human {
		@override
		protected void sayHello() {
			System.out.println("Man Say Hello!");
		}
	}
	static class Woman extends Human {
		@override
		protected void sayHello() {
			System.out.println("Woman Say Hello!");
		}
	}
	public static void main(String[] args) {
		Human man = new Man();
		Human women = new Woman();
		man.sayHello();
		woman.sayHello();
		man = new Woman();
		man.sayHello();
	}
}

這里不是根據(jù)靜態(tài)類型決定的

  • 靜態(tài)類型的Human兩個變量man和woman在調(diào)用sayHello() 方法時執(zhí)行了不同的行為
  • 變量man在兩次調(diào)用中執(zhí)行了不同的方法

導(dǎo)致這個現(xiàn)象的額原因 :這兩個變量的實際類型不同

Java虛擬機(jī)是如何根據(jù)實際類型分派方法的執(zhí)行版本的: 從invokevirtual指令的多態(tài)查找過程開始 ,invokevirtual指令運行時解析過程大致分為以下幾個步驟:

  • 找到操作數(shù)棧頂?shù)牡谝粋€元素所指向的對象的實際類型,記作C
  • 如果在類型C中找到與常量中的描述符和簡單名稱相符合的方法,然后進(jìn)行訪問權(quán)限驗證,如果驗證通過則返回這個方法的直接引用,查找過程結(jié)束;如果驗證不通過,則拋出java.lang.illegalAccessError異常
  • 如果未找到,就按照繼承關(guān)系從下往上依次對類型C的各個父類進(jìn)行第二步的搜索和驗證過程
  • 如果始終沒有找到合適的方法,則拋出java.lang.AbstractMethodError異常

Java語言方法重寫的本質(zhì):

invokevirtual指令執(zhí)行的第一步就是在運行時期確定接收者的實際類型,所以兩次調(diào)用中的invokevirtual指令把常量池中的類方法符號引用解析到了不同的直接引用上

這種在運行時期根據(jù)實際類型確定方法執(zhí)行版本的分派過程就叫做動態(tài)分派

虛擬機(jī)動態(tài)分派的實現(xiàn)

虛擬機(jī)概念解析的模式就是靜態(tài)分派和動態(tài)分派,可以理解虛擬機(jī)在分派中 "會做什么" 這個問題

虛擬機(jī) "具體是如何做到的" 在各種虛擬機(jī)實現(xiàn)上會有差別:

  • 由于動態(tài)分派是非常頻繁的動作,而且動態(tài)分派的方法版本選擇過程需要運行時在類的方法元數(shù)據(jù)中搜索合適的目標(biāo)方法
  • 因此在虛擬機(jī)的實際實現(xiàn)中,為了基于性能的考慮,大部分實現(xiàn)都不會真正的進(jìn)行如此頻繁的搜索
  • 最常用的"穩(wěn)定優(yōu)化"的方式是為類在方法區(qū)中建立一個虛方法表(Virtual Method Table,即vtable), 使用虛方法表索引代替元數(shù)據(jù)查找以提高性能

虛方法表中存放著各個方法的實際入口地址:

  • 如果某個方法在子類中沒有被重寫,那子類的虛方法表里面的地址入口和父類相同方法的地址入口是一致的,都指向父類的實際入口
  • 如果子類中重寫了這個方法,子類方法表中的地址將會替換為指向子類實際方法的入口地址

具有相同簽名的方法,在父類,子類的虛方法表中具有一樣的索引序號:

這樣當(dāng)類型變換時,僅僅需要變更查找的方法表,就可以從不同的虛方法表中按索引轉(zhuǎn)換出所需要的入口地址

方法表一般在類加載階段的連接階段進(jìn)行初始化:

準(zhǔn)備了類的變量初始值后,虛擬機(jī)會把該類的方法表也初始化完畢

以上就是Java方法調(diào)用解析靜態(tài)分派動態(tài)分派執(zhí)行過程的詳細(xì)內(nèi)容,更多關(guān)于Java靜態(tài)動態(tài)分派執(zhí)行過程的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • 聊聊drools?session的不同

    聊聊drools?session的不同

    在drools中存在2種session,一種是有狀態(tài)的Session (Stateful Session),另外一種一種是無狀態(tài)的Session (Stateless Session,本文通過實例代碼給大家介紹drools?session的不同,感興趣的朋友一起看看吧
    2022-05-05
  • Java?IO流之StringWriter和StringReader用法分析

    Java?IO流之StringWriter和StringReader用法分析

    這篇文章主要介紹了Java?IO流之StringWriter和StringReader用法分析,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-12-12
  • JVM自定義類加載器在代碼擴(kuò)展性實踐分享

    JVM自定義類加載器在代碼擴(kuò)展性實踐分享

    這篇文章主要介紹了JVM自定義類加載器在代碼擴(kuò)展性實踐分享,一個類型從被加載到虛擬機(jī)內(nèi)存中開始,到卸載出內(nèi)存為止,它的整個生命周期將會經(jīng)歷加載、驗證、準(zhǔn)備、解析、初始化 、使用和卸載七個階段,其中驗證、準(zhǔn)備、解析三個部分統(tǒng)稱為連接
    2022-06-06
  • java實現(xiàn)圖形卡片排序游戲

    java實現(xiàn)圖形卡片排序游戲

    這篇文章主要為大家詳細(xì)介紹了java實現(xiàn)圖形卡片排序游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-07-07
  • spring Roo安裝使用簡介

    spring Roo安裝使用簡介

    這篇文章主要介紹了spring Roo安裝使用簡介,具有一定借鑒價值,需要的朋友可以參考下
    2017-12-12
  • struts2.5+框架使用通配符與動態(tài)方法常見問題小結(jié)

    struts2.5+框架使用通配符與動態(tài)方法常見問題小結(jié)

    這篇文章主要介紹了struts2.5+框架使用通配符與動態(tài)方法常見問題 ,在文中給大家提到了Struts2.5框架使用通配符指定方法 ,需要的朋友可以參考下
    2018-09-09
  • 詳解Springboot集成sentinel實現(xiàn)接口限流入門

    詳解Springboot集成sentinel實現(xiàn)接口限流入門

    這篇文章主要介紹了詳解Springboot集成sentinel實現(xiàn)接口限流入門,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-11-11
  • JAVA編程不能不知道的反射用法總結(jié)

    JAVA編程不能不知道的反射用法總結(jié)

    這篇文章主要介紹了Java反射技術(shù)原理與用法,結(jié)合實例形式分析了Java反射技術(shù)的基本概念、功能、原理、用法及操作注意事項,需要的朋友可以參考下
    2021-07-07
  • POI XSSFSheet shiftRows bug問題解決

    POI XSSFSheet shiftRows bug問題解決

    這篇文章主要介紹了POI XSSFSheet shiftRows bug問題解決,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-07-07
  • Java中ThreadLocal的使用及原理詳解

    Java中ThreadLocal的使用及原理詳解

    這篇文章主要介紹了Java中ThreadLocal的使用及原理詳解,ThreadLocal是JDK提供的,提供線程本地變量,主要用來存放線程獨有變量和解決參數(shù)傳遞問題的,需要的朋友可以參考下
    2023-09-09

最新評論