Java對(duì)象的內(nèi)存布局詳細(xì)介紹
在HotSpot虛擬機(jī)中,對(duì)象在內(nèi)存中存儲(chǔ)的布局可以分為三塊區(qū)域:對(duì)象頭(Header)、實(shí)例數(shù)據(jù)(Instance Data)和對(duì)齊填充(Padding)。
一、對(duì)象頭
Java對(duì)象的對(duì)象頭由三部分組成:
1)、Mark Word
MarkWord用于存儲(chǔ)對(duì)象自身的運(yùn)行時(shí)數(shù)據(jù), 如哈希碼(HashCode)、GC分代年齡、同步鎖信息、偏向鎖標(biāo)識(shí)等等。Mark Word在32位JVM中的長(zhǎng)度是32bit,在64位JVM中長(zhǎng)度是64bit。
通常我們都是使用的64位的JVM,Mark Word 在64位 JVM 中內(nèi)部結(jié)構(gòu)如下圖:

2)、類型指針
類型指針指向?qū)ο蟮念愒獢?shù)據(jù),虛擬機(jī)通過(guò)這個(gè)指針確定該對(duì)象是哪個(gè)類的實(shí)例。Java對(duì)象的類數(shù)據(jù)保存在方法區(qū)。
3)、數(shù)組長(zhǎng)度(只有數(shù)組對(duì)象才有)
如果對(duì)象是一個(gè)數(shù)組,那么對(duì)象頭還需要有額外的空間用于存儲(chǔ)數(shù)組的長(zhǎng)度。如果對(duì)象是數(shù)組類型,因?yàn)镴VM虛擬機(jī)可以通過(guò)Java對(duì)象的元數(shù)據(jù)信息確定Java對(duì)象的大小,但是無(wú)法從數(shù)組的元數(shù)據(jù)來(lái)確認(rèn)數(shù)組的大小,所以用一塊來(lái)記錄數(shù)組長(zhǎng)度。
二、實(shí)例數(shù)據(jù)
實(shí)例數(shù)據(jù)部分存放類的屬性數(shù)據(jù)信息,包括父類的屬性信息。
通過(guò)示例說(shuō)明每個(gè)區(qū)域具體存放哪些內(nèi)容:
class Student {
private String name;
public Student(String name) {
this.name = name;
}
}
public class Demo {
public static void main(String[] args) {
Student studentA = new Student("zhangsan");
Student studentB = new Student("lisi");
}
}JVM結(jié)構(gòu)圖如下所示:

三、對(duì)齊填充
由于虛擬機(jī)要求對(duì)象起始地址必須是8字節(jié)的整數(shù)倍,所以后面有幾個(gè)字節(jié)用于把對(duì)象的大小補(bǔ)齊至8字節(jié)的整數(shù)倍,沒(méi)有特別的功能,對(duì)齊填充不是必須存在的,僅僅是為了字節(jié)對(duì)齊。
為什么必須是8個(gè)字節(jié)?
根據(jù)“計(jì)算機(jī)組成原理”,8個(gè)字節(jié)是計(jì)算機(jī)讀取和存儲(chǔ)的最佳實(shí)踐。
四、使用JOL工具分析對(duì)象內(nèi)存布局
接下來(lái)我們使用JOL(Java Object Layout)工具,它是一個(gè)用來(lái)分析JVM中Object布局的小工具。包括Object在內(nèi)存中的占用情況,實(shí)例對(duì)象的引用情況等等。
直接在maven工程中加入對(duì)應(yīng)的依賴:
<dependency> <groupId>org.openjdk.jol</groupId> <artifactId>jol-core</artifactId> <version>0.9</version> </dependency>
通過(guò)JOL查看new Object()的對(duì)象布局信息:
public class JOLDemo {
public static void main(String[] args) {
Object obj = new Object();
System.out.println("十進(jìn)制hashCode = " + obj.hashCode());
System.out.println("十六進(jìn)制hashCode = " + Integer.toHexString(obj.hashCode()));
System.out.println("二進(jìn)制hashCode = " + Integer.toBinaryString(obj.hashCode()));
String str = ClassLayout.parseInstance(obj).toPrintable();
System.out.println(str);
}
}運(yùn)行結(jié)果如下:
十進(jìn)制hashCode = 1956725890
十六進(jìn)制hashCode = 74a14482
二進(jìn)制hashCode = 1110100101000010100010010000010
java.lang.Object object internals:
OFFSET SIZE TYPE DESCRIPTION VALUE
0 4 (object header) 01 82 44 a1 (00000001 10000010 01000100 10100001) (-1589345791)
4 4 (object header) 74 00 00 00 (01110100 00000000 00000000 00000000) (116)
8 4 (object header) e5 01 00 f8 (11100101 00000001 00000000 11111000) (-134217243)
12 4 (loss due to the next object alignment)
Instance size: 16 bytes
Space losses: 0 bytes internal + 4 bytes external = 4 bytes total
解釋下各個(gè)字段的含義:
- OFFSET是偏移量,也就是到這個(gè)字段位置所占用的字節(jié)數(shù);
- SIZE是后面類型的大?。?/li>
- TYPE是Class中定義的類型;
- DESCRIPTION是類型的描述;
- VALUE是TYPE在內(nèi)存中的值;
從上圖可以看出Object obj = new Object();在內(nèi)存中占16個(gè)字節(jié),注意最后面的(loss due to the next object alignment)其實(shí)就是對(duì)齊填充的字節(jié)數(shù),這里由于Object obj = new Object();沒(méi)有實(shí)例數(shù)據(jù),對(duì)象頭總共占用了12個(gè)字節(jié)(默認(rèn)開(kāi)啟了指針壓縮-XX:+UseCompressedOops),由于虛擬機(jī)要求對(duì)象起始地址必須是8字節(jié)的整數(shù)倍,所以還需要對(duì)齊填充4個(gè)字節(jié),達(dá)到2倍的8bit。

到此這篇關(guān)于Java對(duì)象的內(nèi)存布局詳細(xì)介紹的文章就介紹到這了,更多相關(guān)Java對(duì)象內(nèi)存布局內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java基礎(chǔ)知識(shí)之StringWriter流的使用
這篇文章主要介紹了Java基礎(chǔ)知識(shí)之StringWriter流的使用,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-12-12
一篇文章帶你復(fù)習(xí)java知識(shí)點(diǎn)
以下簡(jiǎn)單介紹了下我對(duì)于這些java基本知識(shí)點(diǎn)和技術(shù)點(diǎn)的一些看法和心得,這些內(nèi)容都源自于我這些年來(lái)使用java的一些總結(jié),希望能夠給你帶來(lái)幫助2021-06-06
解釋為什么Java中“1000==1000”為false而”100==100“為true
在日常編程中,我們經(jīng)常遇到一些看似簡(jiǎn)單卻隱藏著復(fù)雜邏輯的問(wèn)題,這篇文章主要介紹了解釋為什么Java中“1000==1000”為false而”100==100“為true,需要的朋友可以參考下2024-01-01
一個(gè)Servlet是如何處理多個(gè)請(qǐng)求的?
以前我一直以為一個(gè)Servlet只能處理一個(gè)請(qǐng)求,后來(lái)發(fā)現(xiàn)是自己太菜了,可以借助攜帶一個(gè)參數(shù)來(lái)完成多個(gè)請(qǐng)求的處理,根據(jù)參數(shù)的不同,在核心的service方法中調(diào)用不同的業(yè)務(wù)方法,來(lái)實(shí)現(xiàn)處理多個(gè)servlet請(qǐng)求的目的,廢話不多說(shuō),直接上代碼,需要的朋友可以參考下2021-06-06
Java基礎(chǔ)之ArrayList的擴(kuò)容機(jī)制
這篇文章主要介紹了Java基礎(chǔ)之ArrayList的擴(kuò)容機(jī)制,文中有非常詳細(xì)的代碼示例,對(duì)正在學(xué)習(xí)java基礎(chǔ)的小伙伴們有很好的幫助,需要的朋友可以參考下2021-05-05
spring MVC + bootstrap實(shí)現(xiàn)文件上傳示例(帶進(jìn)度條)
本篇文章主要介紹了spring MVC + bootstrap實(shí)現(xiàn)文件上傳示例(帶進(jìn)度條),非常具有使用價(jià)值,有需要的朋友可以了解一下。2017-03-03

