聊聊JVM虛方法表和方法調用

来源:https://www.cnblogs.com/xiaoniuhululu/archive/2023/07/04/17523404.html
-Advertisement-
Play Games

> 作者:小牛呼嚕嚕 | [https://xiaoniuhululu.com](https://xiaoniuhululu.com/) > 電腦內功、源碼解析、科技故事、項目實戰、面試八股等更多硬核文章,首發於公眾號「[小牛呼嚕嚕](https://www.xiaoniuhululu.com/i ...


作者:小牛呼嚕嚕 | https://xiaoniuhululu.com
電腦內功、源碼解析、科技故事、項目實戰、面試八股等更多硬核文章,首發於公眾號「小牛呼嚕嚕

大家好,我是呼嚕嚕,好久沒更新文章了,今天我們來填個坑,在之前的一篇文章深挖⾯向對象編程三⼤特性 --封裝、繼承、多態
我們遺留了一個問題:當父類引用指向子類對象時,JVM是如何知曉調用的是哪個子類的方法?

動態綁定和靜態綁定

我們下文還是用之前文章的例子,簡單修改一下:

public class ClassTest {

    static class Animal {
        public void eat(){
            System.out.println("動物吃飯!");
        }
        public void work(){
            System.out.println("動物可以幫助人類幹活!");
        }
    }

    static class Cat extends Animal {
        public void eat() {
            System.out.println("吃魚");
        }
        public void sleep() {
            System.out.println("貓會睡懶覺");
        }
    }

    static class Dog extends Animal {
        public void eat() {
            System.out.println("吃骨頭");
        }
    }

    public static void main(String[] args) throws Exception {
        Animal cat=new Cat();
        cat.eat();
        cat.work();
    	  //cat.sleep();//此處編譯會報錯。
    }

}

當父類引用指向子類對象時,也就是Animal cat=new Cat();這個也叫做向上轉型,重寫式多態。

這種多態其實是通過動態綁定(dynamic binding)技術來實現,是指在執行期間判斷所引用對象的實際類型,根據其實際的類型調用其相應的方法。也就是說,只有程式運行起來,你才知道調用的是哪個子類的方法。這種多態可通過函數的重寫以及向上轉型來實現。

與動態綁定相對應的就是靜態綁定,指的是在JVM解析時便能夠直接識別目標方法的情況。網上有些文章說,重載和靜態綁定直接掛鉤,這其實是不完全正確的,筆者舉個極端的例子:當某個類中的重載方法被它的子類重寫時,那它其實通過了動態綁定。

重載指的是方法名相同而參數類型不相同的方法之間的關係,重寫指的是方法名相同並且參數類型也相同的方法之間的關係

需要註意的是:本文一直在說程式在運行期間發生的事,而方法調用在靜態階段(編譯)以聲明的靜態類型為準,不管符號引用指向的是哪個實例對象。編譯成位元組碼再進入JVM,進行類載入

我們回到剛剛的例子上:
cat.eat();這句的結果列印:吃魚。程式這塊調用我們子類Cat定義的方法,而不是父類的同名方法。
cat.work();這句的結果列印:動物可以幫助人類幹活!我們上面Cat類沒有定義work方法,但是卻使用了父類的方法,這是不是很神奇。其實此處調的是父類的同名方法
cat.sleep();這句 編譯器會提示 編譯報錯。表明:當我們當子類的對象作為父類的引用使用時,只能訪問子類中和父類中都有的方法,而無法去訪問子類中特有的方法。雖然向上轉型是安全的。但是缺點是:一旦向上轉型,子類會丟失的子類的擴展方法,其實就是 子類中原本特有的方法就不能再被調用了。所以cat.sleep()這句會編譯報錯。

由此我們可以發現規律:當發生向上轉型,去調用方法時,首先檢查父類中是否有該方法,如果沒有,則編譯錯誤;如果有,再去調用子類的同名方法。如果子類沒有同名方法,會再次去調父類中的該方法。這種根據對象的實際類型而不是聲明類型來選擇並調用方法的過程也叫做動態分派(Dynamic Dispatch)

但如果直接這樣去查找,會發生迴圈查找,效率較低,為瞭解決這個問題,虛方法表 就出現了,也就是動態綁定的底層原理。

虛方法表與虛方法

JVM 虛方法表(Virtual Method Table),也稱為vtable,是動態調度用來依次調用虛方法的一種表結構,是一種特殊的索引表

面向對象編程,會頻繁地觸發動態分派,如果每次動態分配的過程都要重新在類的方法 元數據中搜索合適的目標的方法,就可能影響到執行效率,所以JVM選擇了 用空間換取時間的策略來實現動態綁定,為每個類生成一張虛方法表,然後直接通過虛方法表,使用索引來代替迴圈查找,快速定位目標方法。

類載入器與雙親委派機制一網打盡一文中,我們知道 類的生命周期一般有如下圖有7個階段,其中階段1-5為類載入過程,驗證、準備、解析統稱為連接

虛方法表會在類載入的連接階段被創建,JVM掃描類的方法信息,識別哪些是虛方法,併在虛方法表中儲存其對應的 方法的相關信息以及這些方法在虛擬機記憶體方法區中的入口地址。這入口地址就是該方法的虛擬方法表的索引,JVM可以通過這個索引地址找到對應的方法。也就是說,每個類的對象都會擁有自己的虛方法表

那什麼是虛方法和非虛方法?

非虛方法:如果方法在編譯期就確定了具體的調用版本,則這個版本在運行時是不可變的,這樣的方法稱為非虛方法靜態方法。
比如私有方法,final 方法,實例構造器,父類方法都是非虛方法,除了這些以外都是虛方法

當Java中發生向上轉型,呈現重寫式多態時,如果子類沒有重寫父類方法,子類並不會複製一份父類的方法到自己的虛方法表中,就會去父類的虛方法表中查找 目標方法

子類的重寫的方法和父類中的同名方法在位元組碼層面方法索引通常來說是一樣的,如果在子類找到方法eat(),其索引是0,發現不是要調用的方法後,而是要調用父類的eat(),就會直接去父類方法索引為0的地方查找,這樣能進一步提高查找效率。

JVM方法調用的指令

從JVM底層來瞭解方法調用,我們還需知曉 在JVM中和方法調用有關的指令有5種:

  1. invokeinterface:調用介面中的方法,實際上是在運行期決定的,決定到底調用實現該介面的哪個對象的特定方法。
  2. invokestatic:調用靜態方法。
  3. invokespecial: 調用私有實例方法、構造器方法;使用super關鍵詞調用父類的實例方法、構造器;調用所實現介面的default方法
  4. invokevirtual:調用非私有實例方法,也就是虛方法,運行期動態查找的過程。
  5. invokedynamic: 調用動態方法,JDK7新加入的一個虛擬機指令,相比於之前的四條指令,他們的分派邏輯都是固化在JVM內部,而invokedynamic則用於處理新的方法分派:它允許應用級別的代碼來確定執行哪一個方法調用,只有在調用要執行的時候,才會進行這種判斷,從而達到動態語言的支持。(Invoke dynamic method)

我們javap來反編譯上文例子生成的class文件ClassTest.class:

 public com.zj.ideaprojects.demo.test4.ClassTest();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: return
      LineNumberTable:
        line 3: 0

  public static void main(java.lang.String[]) throws java.lang.Exception;
    descriptor: ([Ljava/lang/String;)V
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=2, locals=2, args_size=1
         0: new           #2                  // class com/zj/ideaprojects/demo/test4/ClassTest$Cat
         3: dup
         4: invokespecial #3                  // Method com/zj/ideaprojects/demo/test4/ClassTest$Cat."<init>":()V
         7: astore_1
         8: aload_1
         9: invokevirtual #4                  // Method com/zj/ideaprojects/demo/test4/ClassTest$Animal.eat:()V
        12: aload_1
        13: invokevirtual #5                  // Method com/zj/ideaprojects/demo/test4/ClassTest$Animal.work:()V
        16: return
      LineNumberTable:
        line 30: 0
        line 31: 8
        line 32: 12
        line 34: 16
    Exceptions:
      throws java.lang.Exception

我們可以發現: Java 中所有非私有實例方法調用都會被編譯成 invokevirtual指令,而介面方法調用都會被編譯成 invokeinterface 指令。這兩種指令,均屬於Java 虛擬機中的虛方法調用,會進行函數的動態綁定。

invokevirtual指令在執行時,首先在運行期確定方法接收者的實際類型,並不是把常量池中方法的符號引用(在這裡相當於常量池裡的方法信息)解析到直接引用上就結束了,而是接著根據方法接收者的實際類型來選擇方法版本,這個過程也就是Java多態的本質。

針對於invokeinterface指令來說,虛擬機會建立一個叫做介面方法表的數據結構(interface method table,簡稱itable),和虛方法表類似。

另外,當我們瞭解invokespecial指令,invokestatic指令時,可以知曉,父類引用在調用靜態方法,私有方法或是介面default方法是不會發生多態,而是直接調用聲明類型的方法。

在Java 8中Lambda表達式和預設方法時,底層會生成和使用invokedynamic,很有意思的一個指令,本文就不詳細介紹該指令了,以後有機會再講講。

小結

小結一下,本文主要講解了方法調用在Java虛擬機的實現方式,以及虛方法表在 JVM 方法調用中充當了一個中介的角色,使得 JVM 能夠實現多態性和動態分派。最後帶大家瞭解一下JVM常見的方法調用的指令,Java可不僅僅只有CRUD哦


參考資料:

https://docs.oracle.com/javase/specs/jvms/se8/html/jvms-2.html#jvms-2.5.2

《Java虛擬機規範》

《深入理解Java虛擬機:JVM高級特性與最佳實踐第3版》


全文完,感謝您的閱讀,如果我的文章對你有所幫助的話,還請點個免費的,你的支持會激勵我輸出更高質量的文章,感謝!

原文鏡像:聊聊JVM虛方法表和方法調用

電腦內功、源碼解析、科技故事、項目實戰、面試八股等更多硬核文章,首發於公眾號「小牛呼嚕嚕」,我們下期再見!


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 線上代碼 a = [1,2,3] def abc(a): a.remove(1) abc(a) print(a) 這段代碼先指定了一個a變數是個list,又寫了一個abc函數,功能是把外面傳進來的list裡面的1這個值去掉 按理說在函數內的執行只應該屬於函數內的變化,但是實際列印結果是[2,3],函 ...
  • 併發指同一時間內進行了多個線程。併發問題是多個線程對同一資源進行操作時產生的問題。通過加鎖可以解決併發問題,ReentrantLock是鎖的一種。 ...
  • **在這篇文章中,我將手把手地教你如何從零開始部署一個使用Django框架的Python服務。無論你是一個剛開始接觸開發的新手,還是一個有經驗的開發者想要快速瞭解Django,這篇教程都會為你提供一條清晰的路徑。我們將從環境搭建開始,一步一步地創建一個可以處理GET和POST請求的服務,讓你能在實踐 ...
  • #### 近日公司有個需求,需要調研如何使用Java來讀取Windows日誌文件(類型:應用程式,安全,Setup,系統) ![](https://img2023.cnblogs.com/blog/1519440/202307/1519440-20230704100117681-1957523520 ...
  • ## 教程簡介 CakePHP是一個運用了諸如ActiveRecord、Association Data Mapping、Front Controller和MVC等著名設計模式的快速開發框架。該項目主要目標是提供一個可以讓各種層次的PHP開發人員快速地開發出健壯的Web應用,而 又不失靈活性。 Ca ...
  • 本文為大家整理彙總了常見的獲取Bean的方式,並提供一些優劣分析,方便大家在使用到時有更好的選擇。同時,也會為大家適當的普及和拓展一些相關知識。 ...
  • ## 教程簡介 DAX代表 Data Analysis Expressions. DAX是一種公式語言,是函數,運算符和常量的集合,可以在公式或表達式中用於計算和返回一個或多個值. DAX是與Excel Power Pivot的數據模型相關聯的公式語言. 它不是一種編程語言,而是一種允許用戶在計算列 ...
  • 學習記錄,不喜勿噴 什麼是OkHttp 一般在Java平臺上,我們會使用Apache HttpClient作為Http客戶端,用於發送 HTTP 請求,並對響應進行處理。比如可以使用http客戶端與第三方服務(如SSO服務)進行集成,當然還可以爬取網上的數據等。OKHttp與HttpClient類似 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...