# Java对象内存布局和对象头

# 对象在堆内存中的布局

在hotspot虚拟机中,对象在堆内存中的存储布局可以分为三个部分:对象头(Header),实例数据(Instance data),对齐填充(Padding)

image-20231017190431419

# 对象头Header

# 对象标记Mark Word:默认存储对象的哈希码,分代年龄和锁标志位等信息

image-20231017191525014 image-20231017191541945

分别定义在==oop.hpp==文件和==markOop.hpp==文件

==GC年龄采用4位bit存储,最大为15==,例如MaxTenuringThreshold参数默认值就是15

markword(64位)分布图,对象布局、GC回收和后面的锁升级就是对象标记MarkWord里面标志位的变化

# 类型指针(类元信息)Klass Pointer:对象指向它的类元数据的指针,虚拟机通过这个指针来确定这个对象是哪个类的实例。

image-20231017190829500

在64位系统中,==Mark Word占了8个字节==,==类型指针占了8个字节==,==Header一共是16个字节==

==如果对象是数组,那么对象头中还有数组长度==

# 实例数据:Instance Data

实例数据存放类的属性(Field)数据信息,包括父类的属性信息, 如果是数组的实例部分还包括数组的长度,这部分内存按4字节对齐。

# 对齐填充:Padding

虚拟机要求对象起始地址必须是8字节的整数倍。填充数据不是必须存在的,仅仅是为了字节对齐这部分内存按8字节补充对齐。

# new Object()

采用官网的jdk工具JOL可以获取对象的信息,分析出对象在jvm中的大小和分布

<dependency>
    <groupId>org.openjdk.jol</groupId>
    <artifactId>jol-core</artifactId>
    <version>0.9</version>
</dependency>
1
2
3
4
5

编写一个demo

public class JOLDemo
{
    public static void main(String[] args)
    {
        Object o = new Object();
        System.out.println( ClassLayout.parseInstance(o).toPrintable());
    }
}
1
2
3
4
5
6
7
8
image-20231017192158144

但是结果表明类型指针为4字节与前文定义不一致,这是因为默认开启了类型指针压缩,以节约空间,可以通过-XX:-UseCompressedClassPointers jvm参数来关闭指针压缩

因此可以得出==new Object()对象占16个字节==

附:Hospot术语表 (opens new window)

C源文件 (opens new window)

​ 深入理解Java虚拟机 48-52页