# Java对象内存布局和对象头
# 对象在堆内存中的布局
在hotspot虚拟机中,对象在堆内存中的存储布局可以分为三个部分:对象头(Header),实例数据(Instance data),对齐填充(Padding)
# 对象头Header
# 对象标记Mark Word:默认存储对象的哈希码,分代年龄和锁标志位等信息
分别定义在==oop.hpp==文件和==markOop.hpp==文件
==GC年龄采用4位bit存储,最大为15==,例如MaxTenuringThreshold参数默认值就是15
markword(64位)分布图,对象布局、GC回收和后面的锁升级就是对象标记MarkWord里面标志位的变化
# 类型指针(类元信息)Klass Pointer:对象指向它的类元数据的指针,虚拟机通过这个指针来确定这个对象是哪个类的实例。
在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
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
2
3
4
5
6
7
8
但是结果表明类型指针为4字节与前文定义不一致,这是因为默认开启了类型指针压缩,以节约空间,可以通过-XX:-UseCompressedClassPointers jvm参数来关闭指针压缩
因此可以得出==new Object()对象占16个字节==
附:Hospot术语表 (opens new window)
深入理解Java虚拟机 48-52页