对象结构
架构师集合之锁+synchronized原理篇
这篇文章中我已经详细说明了对象实例的组成和对象头的组成,这里再讲一下。
对象头:
-
markword:第一部分markword,用于存储对象自身的运行时数据,如哈希码(HashCode)、GC分代年龄、锁状态标志、线程持有的锁、偏向线程ID、偏向时间戳等,这部分数据的长度在32位和64位的虚拟机(未开启压缩指针)中分别为32bit和64bit,官方称它为“MarkWord”。
-
klass :对象头的另外一部分是klass类型指针,即对象指向它的类元数据的指针,虚拟机通过这个指针来确定这个对象是哪个类的实例.
-
数组长度(只有数组对象有): 如果对象是一个数组, 那在对象头中还必须有一块数据用于记录数组长度.
-
实例数据:实例数据部分是对象真正存储的有效信息,也是在程序代码中所定义的各种类型的字段内容。无论是从父类继承下来的,还是在子类中定义的,都需要记录起来。
实际数据:这个区域当然就是描述真实的对象数据。这个区域包括了对象中的所有字段属性信息,它们可能是某个其它对象的地址引用,也可能是基础数据的数据值。
对齐填充:第三部分对齐填充并不是必然存在的,也没有特别的含义,它仅仅起着占位符的作用。由于HotSpot VM的自动内存管理系统要求对象起始地址必须是8字节的整数倍,换句话说,就是对象的大小必须是8字节的整数倍。而对象头部分正好是8字节的倍数(1倍或者2倍),因此,当对象实例数据部分没有对齐时,就需要通过对齐填充来补全。
对象大小计算
-
- 在32位系统下,存放Class指针的空间大小是4字节,MarkWord是4字节,对象头为8字节。
-
- 在64位系统下,存放Class指针的空间大小是8字节,MarkWord是8字节,对象头为16字节。
-
- 64位开启指针压缩的情况下,存放Class指针的空间大小是4字节,MarkWord是8字节,对象头为12字节。因为必须是8字节的整数倍,所以填充4字节,还是16字节。
-
- 静态属性不算在对象大小内。
-
- 属性中数组默认占16字节,然后看初始化数量是多少再相加,比如int[10]就是16+4*10=56字节,这里填充也需要单独算,比如是int[9],那么还是56字节。有一点,String数组占的字节和int一样,是4字节一个。
练习:
class A{
//new A()占多少字节? 64位开启指针压缩的情况下,12(本身)+(4)(int占的字节)=16字节,刚好8的倍数。
private int value=1;
}
class B{
//new B()占多少字节? 16(本身)+(16 + 1*4)(数组不够8的倍数,填充4)=40字节
private int value[]=new int [1];
}
class C{
//new C()占多少字节? 16(本身)+(16 + 8*1)(数组)=40字节
private byte value[]=new byte [8];
}
class D{
//new D()占多少字节? 16(本身)+(16 + 2*4)(数组)=40字节
private String value[]=new String [2];
}
new Object()占多少字节?
这个问题就很简单了,只需要计算对象头大小,32位就是8字节,64位,不管开不开启指针压缩,都是16字节。