漫畫:對象是如何被找到的?句柄 OR 直接指針?
掃描二維碼
隨時隨地手機(jī)看文章
小貼士:想要使用并定位 Java 對象,就要用到 Java 虛擬機(jī)棧(Java Virtual Machine Stack),它描述的是 Java 方法執(zhí)行的線程內(nèi)存模型:每個方法被執(zhí)行的時候,Java 虛擬機(jī)都會同步創(chuàng)建一個棧幀(Stack Frame)用于存儲局部變量表、操作數(shù)棧、動態(tài)連接、方法出口等信息。每一個方法被調(diào)用直至執(zhí)行完畢的過程,就對應(yīng)著一個棧幀在虛擬機(jī)棧中從入棧到出棧的過程。
代碼解讀
以下面代碼為例,來說明對象定位的過程:
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ī)來說, myBus
就是存儲在本地變量表中 reference 類型的變量, new Bus("Java中文社群", "藍(lán)色")
就是存儲在 Java 堆中的對象實(shí)例數(shù)據(jù),它存儲了此實(shí)體類的所有字段信息,例如 code="Java中文社群"
以及 color="藍(lán)色"
等信息,而 Java 堆中的還存儲著對象類型數(shù)據(jù)的地址,它存儲的是對象的類型信息,還有它的父類信息等。
總結(jié)
由于 reference 類型在《Java虛擬機(jī)規(guī)范》里面只規(guī)定了它是一個指向?qū)ο蟮囊?,并沒有定義這個引用應(yīng)該通過什么方式去定位、訪問到堆中對象的具體位置,所以對象訪問方式也是由虛擬機(jī)實(shí)現(xiàn)而定的,主流的訪問方式主要有使用句柄和直接指針兩種:
-
如果使用句柄訪問的話,Java 堆中將可能會劃分出一塊內(nèi)存來作為句柄池,reference 中存儲的就是對象的句柄地址,而句柄中包含了對象實(shí)例數(shù)據(jù)與類型數(shù)據(jù)各自具體的地址信息; -
如果使用直接指針訪問的話,Java 堆中對象的內(nèi)存布局就必須考慮如何放置訪問類型數(shù)據(jù)的相關(guān)信息,reference 中存儲的直接就是對象地址,如果只是訪問對象本身的話,就不需要多一次間接訪問的開銷。
因此使用句柄來訪問的最大好處就是 reference 中存儲的是穩(wěn)定句柄地址,在對象被移動(垃圾收集時移動對象是非常普遍的行為)時只會改變句柄中的實(shí)例數(shù)據(jù)指針,而 reference 本身不需要被修改。使用直接指針訪問速度更快,但如果對象被移動則需要修改 reference 本身。
由于對象訪問在 Java 中非常頻繁,因此這類開銷積少成多也是一項(xiàng)極為可觀的執(zhí)行成本,所以官方默認(rèn)的 HotSpot 虛擬機(jī)采用的就是「直接指針」來定位對象的。
參考 & 鳴謝
周志明《深入理解Java虛擬機(jī)》第 3 版
特別推薦一個分享架構(gòu)+算法的優(yōu)質(zhì)內(nèi)容,還沒關(guān)注的小伙伴,可以長按關(guān)注一下:
長按訂閱更多精彩▼
如有收獲,點(diǎn)個在看,誠摯感謝
免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺僅提供信息存儲服務(wù)。文章僅代表作者個人觀點(diǎn),不代表本平臺立場,如有問題,請聯(lián)系我們,謝謝!