主頁 > 後端開發 > 聊聊JVM虛方法表和方法呼叫

聊聊JVM虛方法表和方法呼叫

2023-07-04 09:47:56 後端開發

作者:小牛呼嚕嚕 | 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虛方法表和方法呼叫

計算機內功、原始碼決議、科技故事、專案實戰、面試八股等更多硬核文章,首發于公眾號「小牛呼嚕嚕」,我們下期再見!

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/556582.html

標籤:Java

上一篇:spring啟動流程 (3) BeanDefinition詳解

下一篇:返回列表

標籤雲
其他(162058) Python(38266) JavaScript(25520) Java(18289) C(15238) 區塊鏈(8275) C#(7972) AI(7469) 爪哇(7425) MySQL(7285) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5876) 数组(5741) R(5409) Linux(5347) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4610) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2438) ASP.NET(2404) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) HtmlCss(1986) .NET技术(1985) 功能(1967) Web開發(1951) C++(1942) python-3.x(1918) 弹簧靴(1913) xml(1889) PostgreSQL(1882) .NETCore(1863) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • 聊聊JVM虛方法表和方法呼叫

    > 作者:小牛呼嚕嚕 | [https://xiaoniuhululu.com](https://xiaoniuhululu.com/) > 計算機內功、原始碼決議、科技故事、專案實戰、面試八股等更多硬核文章,首發于公眾號「[小牛呼嚕嚕](https://www.xiaoniuhululu.com/i ......

    uj5u.com 2023-07-04 09:47:56 more
  • spring啟動流程 (3) BeanDefinition詳解

    BeanDefinition在Spring初始化階段保存Bean的元資料資訊,包括Class名稱、Scope、構造方法引數、屬性值等資訊,本文將介紹一下BeanDefinition介面、重要的實作類,以及在Spring中的使用示例。 # BeanDefinition介面 用于描述了一個Bean實體, ......

    uj5u.com 2023-07-04 09:47:47 more
  • 最簡單的人臉檢測(免費呼叫百度AI開放平臺介面)

    遠程呼叫百度AI開放平臺的web服務,快速完成人臉識別 ### 歡迎訪問我的GitHub > 這里分類和匯總了欣宸的全部原創(含配套原始碼):[https://github.com/zq2599/blog_demos](https://github.com/zq2599/blog_demos) ### ......

    uj5u.com 2023-07-04 09:40:35 more
  • SpringBoot對接阿里云OSS上傳檔案以及回呼(有坑)

    ### 前言 今天在對接阿里云OSS物件存盤, 把這程序記錄下來 ### 鏈接 阿里云的內容很多,檔案是真的難找又難懂 本文主要是用的PostObject API 加上 Callback引數 PostObject -> [https://help.aliyun.com/document_detail ......

    uj5u.com 2023-07-04 07:55:34 more
  • 測驗開發-后端開發do物體類創建

    **創建user表物體類** - 新增do目錄下創建user_entity.py和init.py ``` from sqlalchemy import Integer from server import db """ User表的物體類,與DB欄位一致 """ class UserEntity(d ......

    uj5u.com 2023-07-04 07:55:26 more
  • C++面試八股文:如何實作一個strncpy函式?

    某日二師兄參加XXX科技公司的C++工程師開發崗位第31面: > 面試官:`strcpy`函式使用過吧? > > 二師兄:用過。 > > 面試官:這個函式有什么作用? > > 二師兄:主要用做字串復制,將于字符從一個位置復制到另一個位置。 > > 面試官:`strncpy`函式也使用過吧,和`st ......

    uj5u.com 2023-07-04 07:55:17 more
  • jar-project 代碼加殼加密工具【開源】

    開源地址:https://gitee.com/chejiangyi/jar-protect 介紹 java 本身是開放性極強的語言,代碼也容易被反編譯,沒有語言層面的一些常規保護機制,jar包很容易被反編譯和破解。 受classfinal(已停止維護)設計啟發,針對springboot日常專案開發, ......

    uj5u.com 2023-07-04 07:50:00 more
  • Golang起步篇

    # 一. 安裝Go語言開發環境 ## 1. Wondows下搭建Go開發環境 ### (1). 下載SDK工具包 **sdk下載地址為:**[__https://go.dev/dl/__](https://go.dev/dl/) ![](https://tcs-devops.aliyuncs.com ......

    uj5u.com 2023-07-04 07:44:07 more
  • 解決Springboot專案打成jar包后獲取resources目錄下的檔案報錯的

    前幾天在專案讀取resources目錄下的檔案時碰到一個小坑,明明在本地是可以正常運行的,但是一發到測驗環境就報錯了,說找不到檔案,報錯資訊是:class path resource [xxxx] cannot be resolved to absolute file path because it... ......

    uj5u.com 2023-07-04 07:43:14 more
  • Python web 框架對比:Flask vs Django

    哈嘍大家好,我是咸魚 今天我們從幾個方面來比較一些現在流行的兩個 python web 框架——Flask 和 Django,突出它們的主要特性、優缺點和簡單案例 到最后,大家將更好地了解哪個框架更適合自己的特定需求 參考鏈接:https://djangocentral.com/flask-vs-d ......

    uj5u.com 2023-07-04 07:43:02 more