文章目录
1. 概述
2. 对象头
3. 实例数据
4. 对齐填充
5. 例子(估计对象大小)
1. 概述
一个Java对象在内存中存储的布局可以分为3块区域:对象头(Header)、实例数据(Instance Data)和对齐填充(Padding)
2. 对象头(Header)
HotSpot虚拟机的对象头包括两部分信息,如下所示:
长度 | 内容 | 说明 |
---|---|---|
32/64bit | Mark Word | 包含一系列的标记位,比如轻量级锁的标记位,偏向锁标记位等等。在32位系统占4字节,在64位系统中占8字节 |
32/64bit | Class Pointer | 用来指向对象对应的Class对象(其对应的元数据对象)的内存地址。在32位系统占4字节,在64位系统中(未开启压缩)占8字节 |
32/64bit | Length | 如果是数组对象,还有一个保存数组长度的空间,占4个字节 |
- 第一部分用于存储对象自身的运行时数据,如哈希码(HashCode)、GC分代年龄、锁状态标志、线程持有的锁、偏向线程ID、偏向时间戳、对象分代年龄,这部分信息称为"Mark Word"
- 第二部分是类型指针,用来指向对象对应的Class对象(其对应的元数据对象)的内存地址,虚拟机通过这个指针来确定哪个对象是哪个类的实例
- 如果对象是一个Java数组,那在对象头中还必须有一块用于记录数组长度的数据。因为虚拟机可以通过普通Java对象的元数据信息确定Java对象的大小,但是从数组的元数据中无法确定数组的大小
在 64 位系统及 64 位 JVM 下,开启指针压缩,那么头部存放 Class 指针的空间大小还是4字节,而 Mark Word 区域会变大,变成 8 字节,也就是头部最少为 12 字节
3. 实例数据(Instance Data)
实例数据部分是对象真正存储的有效信息,也是在程序代码中所定义的各种类型的字段内容。
对象实际数据包括了对象的所有成员变量,其大小由各个成员变量的大小决定,比如:byte和boolean是1个字节,short和char是2个字节,int和float是4个字节,long和double是8个字节,reference是4个字节(64位系统中是8个字节)
4. 对齐填充(Padding)
对齐填充不是必然存在的,没有特别的含义,它仅起到占位符的作用。
由于 HotSpot VM 的自动内存管理系统要求对象起始地址必须是 8 字节的整数倍,也就是说对象的大小必须是 8 字节的整数倍。对象头部分是 8 字节的倍数,所以当对象实例数据部分没有对齐时,就需要通过对齐填充来补全。
5. 例子(估计对象大小)
5. 1 创建maven项目导入一个包
(可以查看Java对象内存布局、大小)
<dependency>
<groupId>org.openjdk.jol</groupId>
<artifactId>jol-core</artifactId>
<version>0.9</version>
</dependency>
public class A {
int i;
byte b;
String str;
}
public class T01 {
public static void main(String[] args) {
A a = new A();
System.out.println(ClassLayout.parseInstance(a).toPrintable());
}
}
测试结果
5.2 解释
- 在64位系统下,MarkWord是8字节,存放Class Pointer的空间大小是4字节(默认开启类型指针压缩, -XX:+UseCompressedClassPointers),所以对象头为8+4=12字节
- String 只有引用, 对象引用占用4字节(默认开启了普通对象指针压缩, -XX:+UseCompressedOops)
- byte占用1字节, int占用4字节
- 对象A一共占用了12+4+1+4 = 21字节,按照8字节对其原则,对象大小也就是24字节
在cmd命令查看jvm参数配置信息