Java 对象头

  • Java对象的组成
  • Mark Word 标记字
  • 更直观的理解:


Java对象的组成

java对象 == java 对象头 + 对象体 + 对齐字节(保证是8 byte 的整数倍)

java中文对齐 java字节对齐_后端


对象的几个部分的作用:

  • Mark Word:主要用来表示对象的线程锁状态,另外还可以用来配合GC、存放该对象的hashCode。
  • Klass Word:是一个指向方法区中Class信息的指针,意味着该对象可随时知道自己是哪个Class的实例。
  • 数组长度:占用64位(8字节)的空间,这是可选的,只有当本对象是一个数组对象时才会有这个部分。
  • 对象体:用于保存对象属性和值的主体部分,占用内存空间取决于对象的属性数量和类型。
  • 对齐字节:补齐字节,保证是8 byte 的整数倍,为了减少堆内存的碎片空间。

Mark Word 标记字

在64位机器上,Java对象处于5种不同状态时Mark Word的表现形式,以下每一行代表对象处于某种状态时的样子。

java中文对齐 java字节对齐_开发语言_02


其中各部分的含义如下:

  • biased_lock(1位):偏向锁标识位。由于无锁和偏向锁的锁标识都是 01,没办法区分,引入一位的偏向锁标识位。
  • lock(2位):锁状态标识位。区分锁状态,比如11时表示对象待GC回收状态, 只有最后2位锁标识(11)有效。
  • age(4位):保存对象的分代年龄。表示对象被GC的次数,当该次数到达阈值的时,对象就会转移到老年代。在GC中,如果对象在Survivor区复制一次,年龄增加1。当对象达到设定的阈值时,将会晋升到老年代。默认情况下,并行GC的年龄阈值为15,并发GC的年龄阈值为6。由于age只有4位,所以最大值为15,这就是-XX:MaxTenuringThreshold选项最大值为15的原因。
  • identity_hashcode:保存对象的哈希码。运行期间调用System.identityHashCode()来计算,延迟计算,并把结果赋值到这里。当对象加锁后(偏向、轻量级、重量级),MarkWord的字节没有足够的空间保存hashCode,因此该值会移动到管程Monitor中。
  • thread:保存持有偏向锁的线程ID。偏向模式的时候,当某个线程持有对象的
    时候,对象这里就会被置为该线程的ID。 在后面的操作中,就无需再进行尝试获取锁的动作。这个线程ID并不是JVM分配的线程ID号,和Java Thread中的ID是两个概念。
  • epoch:保存偏向时间戳。偏向锁在CAS锁操作过程中,偏向性标识,表示对象更偏向哪个锁。
  • ptr_to_lock_record:轻量级锁状态下,指向栈中锁记录的指针。当锁获取是无竞争时,JVM使用原子操作而不是OS互斥,这种技术称为轻量级锁定。在轻量级锁定的情况下,JVM通过CAS操作在对象的Mark Word中设置指向锁记录的指针
  • ptr_to_heavyweight_monitor:重量级锁状态下,指向对象监视器Monitor的指针。如果两个不同的线程同时在同一个对象上竞争,则必须将轻量级锁定升级到Monitor以管理等待的线程。在重量级锁定的情况下,JVM在对象的ptr_to_heavyweight_monitor设置指向Monitor的指针。

更直观的理解:

锁状态

存储内容

偏向锁标识位

锁标识位

无锁

哈希码、GC分代年龄

0

01

偏向锁

线程ID、时间戳、GC分代年龄

1

01

轻量级锁

指向栈中锁记录的指针ptr


00

重量级锁

指向Monitor的指针ptr


10

GC标记



11