自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

從hotspot源碼層面剖析Java的多態(tài)實(shí)現(xiàn)原理

開發(fā) 后端
多態(tài)需要通過(guò)動(dòng)態(tài)綁定才能得以實(shí)現(xiàn),而綁定通俗一點(diǎn)講就是讓不同的對(duì)象對(duì)同一個(gè)函數(shù)進(jìn)行調(diào)用,或者反過(guò)來(lái)講,就是讓同一個(gè)函數(shù)與不同的對(duì)象綁定起來(lái),所以多態(tài)得以實(shí)現(xiàn)的一個(gè)大前提就是,編程語(yǔ)言必須是面向?qū)ο蟮摹?/div>

[[393260]]

由于多態(tài)需要通過(guò)動(dòng)態(tài)綁定才能得以實(shí)現(xiàn),而綁定通俗一點(diǎn)講就是讓不同的對(duì)象對(duì)同一個(gè)函數(shù)進(jìn)行調(diào)用,或者反過(guò)來(lái)講,就是讓同一個(gè)函數(shù)與不同的對(duì)象綁定起來(lái),所以多態(tài)得以實(shí)現(xiàn)的一個(gè)大前提就是,編程語(yǔ)言必須是面向?qū)ο蟮?。同時(shí),函數(shù)與對(duì)象相互綁定,意味著函數(shù)也屬于對(duì)象的一部分,這便具備了封裝的特性。因?yàn)橛辛朔庋b,才有了對(duì)象。同時(shí),一個(gè)函數(shù)能夠綁定多個(gè)對(duì)象,意味著對(duì)各不同的對(duì)象具有相同的行為,這是繼承的含義。 因此,面向?qū)ο蟮娜筇匦匀币徊豢?。封裝與繼承其實(shí)是為了多態(tài)準(zhǔn)備的,或者說(shuō),封裝與繼承成全了多態(tài),多態(tài)讓封裝與繼承的意義最大化。

C++是如何實(shí)現(xiàn)多態(tài)的

多態(tài)的實(shí)現(xiàn),現(xiàn)在幾乎所有的編程語(yǔ)言都是基于虛表實(shí)現(xiàn)的,英文vtable。這里我沒(méi)有說(shuō)全部,因?yàn)槲乙膊皇撬械恼Z(yǔ)言都了解哈,不敢亂說(shuō),免得遭噴。^_^

C++的虛表在哪呢?在new創(chuàng)建的對(duì)象的頭部。虛表里面存儲(chǔ)的是什么呢?是虛函數(shù)。C++這塊的知識(shí)我就不講太多了,很多小伙伴不了解C++,講多了沒(méi)必要,作為一名Java程序員,了解到這個(gè)程度夠了。

因?yàn)閔otshot主要是用C++寫的,講了C++的虛表,這張圖你應(yīng)該就能看懂了。

不然總有小伙伴問(wèn)我:Java的類對(duì)應(yīng)的C++對(duì)象,為什么有C++級(jí)別的虛表啊。我沒(méi)看到哪里有這樣的代碼啊。

搞清楚了虛表,再來(lái)了解虛表分發(fā)就容易多了。虛表分發(fā),其實(shí)就是通過(guò)虛表內(nèi)存地址拿到虛表記錄,然后通過(guò)函數(shù)名+內(nèi)含參數(shù)信息及返回值信息的簽名去虛表中找。因?yàn)槭菑那巴笳?,所以如果子類重寫了父類的方法,?huì)調(diào)用子類的方法。C++的虛表分發(fā),我只是簡(jiǎn)單講了下,講多了大家沒(méi)概念。JVM的虛表分發(fā),我等下會(huì)講得詳細(xì)一些。很多現(xiàn)象,如果不了解它的底層,是不是百思不得其解。有那么多為什么?為什么?^_^

所以Java雖好,底層也很重要。順便說(shuō)下,虛表就是用數(shù)組實(shí)現(xiàn)的,沒(méi)有有些小伙伴想得那么復(fù)雜。

JVM中的虛表

JVM的虛表跟C++的虛表還不太一樣。不一樣體現(xiàn)在哪呢?研究虛表研究三個(gè)東西:虛表在哪、虛表是用什么結(jié)構(gòu)實(shí)現(xiàn)的、虛表分發(fā)機(jī)制是怎樣的。JVM的虛表分發(fā)等下講,JVM的虛表也是用數(shù)組實(shí)現(xiàn)的,那這個(gè)不一樣就體現(xiàn)在虛表在哪?

Java的類,JVM中對(duì)應(yīng)的C++對(duì)象是klass模型。Java的對(duì)象,JVM中對(duì)應(yīng)的C++對(duì)象是oop模型。C++中的虛表在對(duì)象頭中,而JVM的虛表在klass模型的頭部,即Java類對(duì)象的頭部。這點(diǎn)區(qū)別一定要記住,這樣你才能理解Java對(duì)象的內(nèi)存布局。

問(wèn)個(gè)問(wèn)題:我們隨便定義的一個(gè)類,它有沒(méi)有JVM虛表呢?其實(shí)是有的。那是哪些方法的內(nèi)存地址呢?回答這個(gè)問(wèn)題前先得搞明白:什么樣的方法會(huì)存入虛表。只有public、protect類型的,且不被static、final修飾的方法才能被多態(tài)調(diào)用,才會(huì)進(jìn)入虛表。因?yàn)镴ava中所有的類都是Object的子類,所以O(shè)bject中滿足這個(gè)條件的方法都會(huì)在每個(gè)類的虛表中。

又到了小伙伴不服氣環(huán)節(jié)。沒(méi)事,上證據(jù)。具體怎么查看我就不講了,有點(diǎn)復(fù)雜。對(duì)hotspot沒(méi)一定的功力講了也沒(méi)概念。

Java是如何實(shí)現(xiàn)虛表分發(fā)

有些小伙伴不理解:我只會(huì)Java干活都沒(méi)問(wèn)題呀,我為什么要學(xué)底層呢?那你想進(jìn)大廠跟優(yōu)秀的人成為同事嗎?你想成為別人眼中的大佬嗎?你希望在某個(gè)領(lǐng)域能有一定的名氣嗎……這些都需要實(shí)力來(lái)支撐。

有些小伙伴說(shuō):我手寫一個(gè)JVM干什么呢?那我就用我手寫的JVM來(lái)講解這個(gè)知識(shí)點(diǎn)。這就是你有一個(gè)手寫JVM的意義之一。

JVM實(shí)現(xiàn)虛表分發(fā),對(duì)應(yīng)的字節(jié)碼指令有兩個(gè):invokevirtual、invokeinterface。上篇文章咱們深入講解了invokeinterface,這篇文章咱們繼續(xù)拿這個(gè)指令來(lái)講這個(gè)知識(shí)點(diǎn)。我們來(lái)看看JVM是如何分發(fā)的。其實(shí)一看執(zhí)行invokeinterface時(shí)的堆棧,你應(yīng)該就能明白了。

雖然invokeinterface后面的操作數(shù)是接口方法信息。但是真正的對(duì)象會(huì)作為this傳過(guò)來(lái)。所以在調(diào)用的時(shí)候,從操作數(shù)棧拿到真正的對(duì)象,然后通過(guò)對(duì)象頭中的類型指針拿到TestDuotai對(duì)應(yīng)的C++類對(duì)象,即klass模型。前面說(shuō)了,虛表就在這個(gè)對(duì)象的頭部。然后通過(guò)函數(shù)名+內(nèi)含參數(shù)信息及返回值信息的簽名去虛表中找。因?yàn)槭菑那巴笳?,所以如果子類重寫了父類的方法,?huì)調(diào)用子類的方法。這就是JVM虛表分發(fā)的底層原理。

 

責(zé)任編輯:姜華 來(lái)源: 今日頭條
相關(guān)推薦

2021-07-20 10:26:53

源碼底層ArrayList

2024-01-19 12:48:00

Redis存儲(chǔ)數(shù)據(jù)庫(kù)

2020-09-02 07:03:04

虛擬機(jī)HotSpotJava

2022-12-07 11:21:30

Reactdiff

2023-11-16 09:10:18

多態(tài)封裝繼承

2021-06-30 10:32:33

反射多態(tài)Java

2019-11-25 16:05:20

MybatisPageHelpeJava

2021-03-06 22:41:06

內(nèi)核源碼CAS

2018-10-31 15:54:47

Java線程池源碼

2017-03-22 14:23:58

Java HashMa實(shí)現(xiàn)原理

2024-01-29 15:54:41

Java線程池公平鎖

2025-03-14 12:30:00

Redis RDBRedis數(shù)據(jù)庫(kù)

2022-07-14 08:02:57

Netty網(wǎng)絡(luò)模塊

2021-03-15 09:44:39

Broker源碼RocketMQ

2021-05-26 11:30:24

Java線程池代碼

2015-03-23 10:04:43

c++編譯器c++實(shí)現(xiàn)原理總結(jié)

2023-11-15 18:53:06

線程客戶端

2013-10-10 14:52:53

jQueryDeferred

2010-09-14 09:30:04

Java多態(tài)

2021-09-10 00:34:22

Java 線程啟動(dòng)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)