成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Java 的多態在 JVM 里原來是這樣的

云計算 虛擬化
面向對象的編程語言里,「多態」是一個至關重要的概念。我們常說,面向對象的本質,是方法與數據的綁定。那對于一個擁有繼承關系的類之間,方法的綁定,是終是子類「重寫」父類的方法,通過父類的引用指向子類的對象,實現運行時的多態。

 多態

面向對象的編程語言里,「多態」是一個至關重要的概念。我們常說,面向對象的本質,是方法與數據的綁定。那對于一個擁有繼承關系的類之間,方法的綁定,是終是子類「重寫」父類的方法,通過父類的引用指向子類的對象,實現運行時的多態。

[[327572]]

說起來比較繞,我們先以僅次于Hello World 著名的 「動物 - 狗」代碼來說明多態,然后再來分析在 JVM 層面,多態是怎樣實現的。

  1. package com.example.demo; 
  2. public class Demo { 
  3.     public static void main(String[] args) { 
  4.         Animal a = new Animal(); 
  5.         a.say(); 
  6.         Dog d = new Dog(); 
  7.         d.say(); 
  8.         Animal ad = new Dog(); 
  9.         ad.say(); 
  10.     } 
  11. class Animal { 
  12.     public void say() { 
  13.         System.out.println("Animal say"); 
  14.     } 
  15.     public void play() { 
  16.         System.out.println("play..."); 
  17.     } 
  18. class Dog extends Animal { 
  19.     public void say() { 
  20.         System.out.println("Dog say"); 
  21.     } 

輸出的內容對于習慣了面向對象的 Java 開發者來說都比較熟悉

  1. Animal say 
  2. Dog say 
  3. Dog say 

那虛擬機是怎樣知道到底要調用 Animal 的 say 還是 Dog 的say呢?

咱們從字節碼的層面來看一下。

  1. 0 new #2 <com/example/demo/Animal> 
  2.  3 dup 
  3.  4 invokespecial #3 <com/example/demo/Animal.<init>> 
  4.  7 astore_1 
  5.  8 aload_1 
  6.  9 invokevirtual #4 <com/example/demo/Animal.say> 
  7. 12 new #5 <com/example/demo/Dog> 
  8. 15 dup 
  9. 16 invokespecial #6 <com/example/demo/Dog.<init>> 
  10. 19 astore_2 
  11. 20 aload_2 
  12. 21 invokevirtual #7 <com/example/demo/Dog.say> 
  13. 24 new #5 <com/example/demo/Dog> 
  14. 27 dup 
  15. 28 invokespecial #6 <com/example/demo/Dog.<init>> 
  16. 31 astore_3 
  17. 32 aload_3 
  18. 33 invokevirtual #4 <com/example/demo/Animal.say> 
  19. 36 return 

你發現沒有,在字節碼的第9行,和第33行,分別對應到 d.say() 和 ad.say() ,但指令內容其實是一樣的。這就神奇了。

在這兩個方法執行前,第8行和第32行,會有一個aload的操作,是把這兩個對象的引用 壓到棧頂,給后面的操作用。這兩個對象,一般也被稱為方法的接收者(Receiver),如果熟悉 Golang等語言的朋友,對這個概念也不陌生。

從9行和第33行看,無論是方法調用的字節碼指令還是參數,都指向了常量池的第4項。都是一樣的,但最終結果并不相同。這里的重點在于 invokevirtual 這個指令的多態指行查找過程,即根據對象的 vtable 在運行時定位方法。

啥是 vtable?

前面的內容提到指令執行時從棧頂獲取當前方法的「接收者」,通過invokerirtual 來執行這個接者者對應的方法。 注意這里的 virtual,和C++的虛方法類似。這個咱們不提,只說Java 的。

對象都有一個自己的「方法表」,這個表里除了自己的方法,還有從父類繼承來的方法,甚至重寫的父類的方法。所以,對應于重寫與重載,體現在方法表里也有所區別。每個子類繼承父類的時候,都將直接復制一份父類的方法表,而對于父類方法的重寫,會直接更新方法表里相同順序的這個方法。

而重載,本質上由于簽名及參數的區別,是一個新的方法,在方法表里會是新增一個元素。

這里的這個方法表,就是咱們說的 vtable(Virtual Method Table),表里的每個方法,對應的是它的實際執行入口地址。如果沒有重寫,那父類和子類的地址是一樣的,都指向父類的實現。

如果子類重寫之后,子類方法表里的這個方法的地址就指向了自己實現的版本。

而我們上面字節碼處觀察到的,兩個 invokevirtual 對應的常量池索引序號是一樣的,這樣實現對于變換實現類型時,查找方法表只需要換個對象,索引依舊相同。

觀察

為了便于 Attach 到 Java 進程,可以在代碼里加下 latch 進行 awiat 阻塞,啟動 SA 就能觀察了。

 

選擇 ClassBrowser

 

在 Class列表里就能找到咱們上面創建的對象。@ 符號后面是這個對象對應的內存地址。復制上Dog的地址,再從菜單里選擇Inspector,

你看 _vtable_len: 7

 

這是告訴我們 vtable 長度是7,里面有7個方法。

實際上咱們在這個類里只重寫了父類 Animal 的 say方法,其它的是從 Animal 繼承來的 play方法,以及超類 Object 里的 5個方法,大概這個樣子

 

JVM 在首次加載類的時候,會解析類內包含的方法,方法解析之后就會計算當前類 vtable的大小。

可能你會問,Object 類內不止5個方法,為什么只算5個呢?而且我們新增其它static、 final 這一類的方法呢?

這里 vtable 只計算非static final 的,全部計算完就得出了vtable_len這個值。

每個 Java 的 Class 在 JVM 內部都會有一個自己的instanceKlass, vtable就分配在這個的最后。

整個instanceKlass的大小,在64位系統里大小是 0x1b8,記住它,后面用的著。 所以咱們上面看到了Dog 類的內存地址,繼續找就能看到他其它方法對應的內存地址。

在Windows -> console 里執行這個:

  1. mem 0x7C0060DD0 7 

這個值怎么來的呢?是從對象的內存地址開始,加上 instanceKlass的大小。

  1. 0x7C0060DD0   =  0x00000007c0060c18 + 0x1b8 

由于我們有7個方法,所以順序查找7個地址。

所以你應該也發現了,Java 里對應這種重寫的方法,是在類加載的時候,才能知道具體對應的是哪個方法,因此也被稱為動態綁定或者遲綁定。

總結起來,這里的 vtable,相當于你的工具清單,有什么能力都做了羅列,像鋼鐵俠的各項技能,每個功能指向具體的超能力,在我們代碼里可以把它理解成一個數組,數組的每個元素指向一個方法地址。

感興趣的話,你加個static 的方法自己找找,看看在不在這里面呢?畢竟static方法執行不是有 invokestatic 指令嘛。

本文轉載自微信公眾號「 Tomcat那些事兒」,可以通過以下二維碼關注。轉載本文請聯系 Tomcat那些事兒公眾號。

 

責任編輯:武曉燕 來源: Tomcat那些事兒
相關推薦

2022-12-14 07:32:40

InnoDBMySQL引擎

2018-04-02 15:13:21

網絡

2023-02-15 08:17:38

2024-04-30 08:22:51

Figma圖形編輯變換矩陣

2025-02-17 09:22:16

MySQLSQL語句

2014-07-21 10:32:52

蘋果公司實習

2022-05-05 08:55:12

工業物聯網IIoT

2024-02-06 09:30:25

Figma矩形矩形物理屬性

2023-05-22 15:58:11

2016-10-12 08:54:24

2020-11-24 06:20:02

Linux日志文件系統

2021-08-17 07:00:00

雙重檢查鎖Nacos

2022-07-13 10:37:59

服務器故障優化

2015-03-25 09:55:34

程序員程序員修補BUG真正原因

2018-10-26 10:41:19

2020-03-23 08:30:12

程序員男友感受

2017-06-06 15:13:07

2009-03-10 12:42:45

2022-08-15 08:01:00

三色標記JVM算法

2021-02-07 08:13:18

@DateTimeFo@NumberFormSpring
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 欧美激情视频一区二区三区免费 | 99久久久无码国产精品 | 日本一区二区三区免费观看 | 九九热re | 超碰婷婷 | 国产极品粉嫩美女呻吟在线看人 | 综合激情久久 | 亚洲精品在线播放 | 精品香蕉一区二区三区 | 91社区在线观看高清 | 一区二区在线不卡 | 亚洲a视| 久久久久久国产精品 | 本道综合精品 | 91av在线视频观看 | 日韩高清成人 | 久久久毛片 | av片毛片 | 欧美成人一区二区三区 | 综合网视频 | 午夜亚洲 | 欧美高清dvd | av不卡一区 | 狠狠操你| 性色av一区二区三区 | 中文字幕日韩一区二区 | 欧美国产日韩一区 | 国产精品18久久久 | 欧美a在线看 | 91欧美精品 | 粉嫩一区二区三区四区公司1 | 免费在线看a | 欧美一级在线观看 | 国产精品人人做人人爽 | 久久国产精品久久久久 | 国产精品九九九 | 在线中文视频 | 日本精品一区二区三区视频 | 成人免费毛片在线观看 | h视频在线免费观看 | 久久久亚洲精品视频 |