JUC并发编程第十一篇,Java对象的内存布局

JUC并发编程第十一篇,Java对象的内存布局

    • 一、对象在堆内存中的存储布局
      • 1、对象头
        • 对象标记Mark Word
        • 类元信息(类型指针)
      • 2、实例数据
      • 3、对齐填充
    • 二、对象标记(MarkWord)布局与验证
      • 代码验证(JOL)

一、对象在堆内存中的存储布局

  • 在 HotSpot 虚拟机里,对象在堆内存中的存储布局可以划分为三个部分:对象头、实例数据、对齐填充(对齐填充保证8个字节的倍数)。
    JUC并发编程第十一篇,Java对象的内存布局_第1张图片

1、对象头

对象头分为对象标记(markOop)和类元信息(klassOop),类元信息存储的是指向该对象类元数据(klass)的首地址。

对象标记Mark Word

对象标记中都保存了什么?如下:
JUC并发编程第十一篇,Java对象的内存布局_第2张图片
JUC并发编程第十一篇,Java对象的内存布局_第3张图片

  • 对象标记默认存储对象的 HashCode、分代年龄 和 锁标志位等信息。
  • 这些信息都是与对象自身定义无关的数据,所以 MarkWord 被设计成一个非固定的数据结构以便在极小的空间内存存储尽量多的数据。
  • 它会根据对象的状态复用自己的存储空间,在运行期间 MarkWord 里存储的数据会随着锁标志位的变化而变化。

类元信息(类型指针)

对象指向它的类元数据的指针,虚拟机通过这个指针来确定这个对象是哪个类的实例。
JUC并发编程第十一篇,Java对象的内存布局_第4张图片
在64位系统中,Mark Word占了8个字节,类型指针占了8个字节,一共是16个字节。

2、实例数据

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

3、对齐填充

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

二、对象标记(MarkWord)布局与验证

Hotspot实现(64位)
JUC并发编程第十一篇,Java对象的内存布局_第5张图片

代码验证(JOL)

  • 添加下面JOL依赖
<dependency>
    <groupId>org.openjdk.jolgroupId>
    <artifactId>jol-coreartifactId>
    <version>0.9version>
dependency>
  • 打印详细信息
public class MyObject {
    public static void main(String[] args) {
        //VM的详细情况
        System.out.println(VM.current().details());
        //所有的对象分配的字节都是8的整数倍。
        System.out.println(VM.current().objectAlignment());
    }
}

JUC并发编程第十一篇,Java对象的内存布局_第6张图片

public class MyObject {
    public static void main(String[] args) {
        Object o = new Object();
        System.out.println( ClassLayout.parseInstance(o).toPrintable());
    }
}

JUC并发编程第十一篇,Java对象的内存布局_第7张图片

  • 对象里边添加字段试试
public class MyObject {
    public static void main(String[] args) {

        System.out.println( ClassLayout.parseInstance(new O()).toPrintable());
    }
}
class O{
    private int i;
    private boolean flag;
    private double d;
}

JUC并发编程第十一篇,Java对象的内存布局_第8张图片

你可能感兴趣的:(JUC并发编程,java,jvm,数据结构,JUC并发编程)