Java对象的内存分布(二)
在Java对象的内存分布(一)中我们简单介绍了Java对象的内存概览,本文将继续深入探讨其它细节。
1、Java对象实例内存布局
指针压缩:java7之后, 堆内存分配超过32GB,指针压缩失效,采用8字节存储引用,否则引用采用4字节存储,更多详情参阅 Java 引用是4个字节还是8个字节?
2、Java对象头——MarkWord
32位机器markword如下图所示;
64 位机器中,GC年龄,偏向锁标识,锁标识均不变,前面的其它标识进行了位数扩充。
- HashCode 指的是
identityHashCode
,详见JAVA中的hashcode。- GC年龄4bit,所以最大为15岁;对象被判定为不可达,GC标识为
11
;更多GC详见细说 Java GC 垃圾收集器。- 内置锁
synchronized
随共享资源竞争激烈程度依次升级,更多详情参见JAVA 锁—— synchronized。
3、Java对象头——Klass pointer
Klass Pointer:对象指向它的类元数据的指针,虚拟机通过这个指针来确定这个对象是哪个类的实例。
Klass pointer :不要怀疑,这里就是Klass,区分java语言中关键字
class
,因为该指针指向的对象在方法区,且为C(C++)语言对象,我们常说的class
对象在堆中,是方法区中Klass对象提供给java对象访问的入口而已。更多详情参见JVM层面的JAVA类和实例(Klass-OOP)。
4、Java对象头——数组长度
数组长度类型为int
,理论上最大长度为 java.lang.Integer.MAX_VALUE = 2^31-1 = 2147483647。
- 实际受到具体的实现带来的限制,这会使得实际的JVM不一定能支持上面说的理论上的最大length。
- 对于元素类型不同的数组,实际能创建的数组的最大length也会不同。
上图在64位JVM上实验,未开启指针压缩,所以引用长度为4Byte。