當(dāng)前位置:首頁(yè) > 公眾號(hào)精選 > 架構(gòu)師社區(qū)
[導(dǎo)讀]小貼士:想要使用并定位 Java 對(duì)象,就要用到 Java 虛擬機(jī)棧(Java Virtual Machine Stack),它描述的是 Java 方法執(zhí)行的線程內(nèi)存模型:每個(gè)方法被執(zhí)行的時(shí)候,Java 虛擬機(jī)都會(huì)同步創(chuàng)建一個(gè)棧幀(Stack Frame)用于存儲(chǔ)局部變量表、操作數(shù)棧、動(dòng)態(tài)連接、方法

漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?

漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?

漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?

小貼士:想要使用并定位 Java 對(duì)象,就要用到 Java 虛擬機(jī)棧(Java Virtual Machine Stack),它描述的是 Java 方法執(zhí)行的線程內(nèi)存模型:每個(gè)方法被執(zhí)行的時(shí)候,Java 虛擬機(jī)都會(huì)同步創(chuàng)建一個(gè)棧幀(Stack Frame)用于存儲(chǔ)局部變量表、操作數(shù)棧、動(dòng)態(tài)連接、方法出口等信息。每一個(gè)方法被調(diào)用直至執(zhí)行完畢的過(guò)程,就對(duì)應(yīng)著一個(gè)棧幀在虛擬機(jī)棧中從入棧到出棧的過(guò)程。

代碼解讀

以下面代碼為例,來(lái)說(shuō)明對(duì)象定位的過(guò)程:

class Bus extends Car {
    private String code;
    private String color;
    Bus(String code, String color) {
        this.code = code;
        this.color = color;
    }
    // 省略其他方法...
}
public class ReferenceTest {
    Bus myBus = new Bus("Java中文社群""藍(lán)色");
}

以官方默認(rèn)的 HotSpot 虛擬機(jī)來(lái)說(shuō), myBus 就是存儲(chǔ)在本地變量表中 reference 類型的變量, new Bus("Java中文社群", "藍(lán)色") 就是存儲(chǔ)在 Java 堆中的對(duì)象實(shí)例數(shù)據(jù),它存儲(chǔ)了此實(shí)體類的所有字段信息,例如 code="Java中文社群" 以及 color="藍(lán)色" 等信息,而 Java 堆中的還存儲(chǔ)著對(duì)象類型數(shù)據(jù)的地址,它存儲(chǔ)的是對(duì)象的類型信息,還有它的父類信息等。

總結(jié)

由于 reference 類型在《Java虛擬機(jī)規(guī)范》里面只規(guī)定了它是一個(gè)指向?qū)ο蟮囊?,并沒(méi)有定義這個(gè)引用應(yīng)該通過(guò)什么方式去定位、訪問(wèn)到堆中對(duì)象的具體位置,所以對(duì)象訪問(wèn)方式也是由虛擬機(jī)實(shí)現(xiàn)而定的,主流的訪問(wèn)方式主要有使用句柄和直接指針兩種:

  • 如果使用句柄訪問(wèn)的話,Java 堆中將可能會(huì)劃分出一塊內(nèi)存來(lái)作為句柄池,reference 中存儲(chǔ)的就是對(duì)象的句柄地址,而句柄中包含了對(duì)象實(shí)例數(shù)據(jù)與類型數(shù)據(jù)各自具體的地址信息;
  • 如果使用直接指針訪問(wèn)的話,Java 堆中對(duì)象的內(nèi)存布局就必須考慮如何放置訪問(wèn)類型數(shù)據(jù)的相關(guān)信息,reference 中存儲(chǔ)的直接就是對(duì)象地址,如果只是訪問(wèn)對(duì)象本身的話,就不需要多一次間接訪問(wèn)的開(kāi)銷。

因此使用句柄來(lái)訪問(wèn)的最大好處就是 reference 中存儲(chǔ)的是穩(wěn)定句柄地址,在對(duì)象被移動(dòng)(垃圾收集時(shí)移動(dòng)對(duì)象是非常普遍的行為)時(shí)只會(huì)改變句柄中的實(shí)例數(shù)據(jù)指針,而 reference 本身不需要被修改。使用直接指針訪問(wèn)速度更快,但如果對(duì)象被移動(dòng)則需要修改 reference 本身。

由于對(duì)象訪問(wèn)在 Java 中非常頻繁,因此這類開(kāi)銷積少成多也是一項(xiàng)極為可觀的執(zhí)行成本,所以官方默認(rèn)的 HotSpot 虛擬機(jī)采用的就是「直接指針」來(lái)定位對(duì)象的。

參考 & 鳴謝

周志明《深入理解Java虛擬機(jī)》第 3 版

    

特別推薦一個(gè)分享架構(gòu)+算法的優(yōu)質(zhì)內(nèi)容,還沒(méi)關(guān)注的小伙伴,可以長(zhǎng)按關(guān)注一下:

漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?

長(zhǎng)按訂閱更多精彩▼

漫畫(huà):對(duì)象是如何被找到的?句柄 OR 直接指針?

如有收獲,點(diǎn)個(gè)在看,誠(chéng)摯感謝

免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺(tái)僅提供信息存儲(chǔ)服務(wù)。文章僅代表作者個(gè)人觀點(diǎn),不代表本平臺(tái)立場(chǎng),如有問(wèn)題,請(qǐng)聯(lián)系我們,謝謝!

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
關(guān)閉
關(guān)閉