一、准备
- 环境:
Java(TM) SE Runtime Environment (build 1.8.0_212-b10)
Java HotSpot(TM) 64-Bit Server VM (build 25.212-b10, mixed mode)
- Maven依赖:jol-core(version:0.10)
二、定义&参考
- 说明:
对象的内存以字节为单位,必须是8的倍数(压缩由去末尾3个0,恢复则补零),它的构成由3部分组成:对象头+实例数据+对齐内存。对象头主 要包括对象的运行行元数据,比较哈希码、GC分代年龄、锁状态标志还有类型指针,类型指针指向类元数据,表明该对象所属类型。实例数据包括自身数据和所有父级数据,所有父级占内存大小都是8的倍数,没有就需要补齐。类型指针一般为4字节,在关闭压缩普通对象指针时(-XX:+UseCompressedOops)为8字节,UseCompressedOops默认是开启的,只有虚拟机内存达到32G以上,4个字节已经无法满足寻址需求时,才需要关闭该参数。
普通对象头除类型指针外的大小为8字节,在开启压缩总大小为12字节,不开启压缩总大小为16字节;数组对象头(数组对象头也有对象填充)在开启压缩时是16字节,不开启压缩为24字节。
- 各种类型大小
对象类型
字节
boolean
1
byte
1
short
2
char
2
int
4
float
4
long
8
double
8
引用类型
开启指针压缩为4,不开启为8
普通对象头
开启指针压缩为12,不开启为8
数据对象头
开启指针压缩为16,不开启为24
三、示例说明(演示用jol-col,实际也可用Java工具:Java VisualVM)
- 示例1:
public class Person { private int id; }
开启压缩普通对象指针时,对象大小:(8+4)+4+0=16字节,补齐0字节
关闭压缩普通对象指针时,对象大小:(8+8)+4+4=24字节,补齐4字节
/** * 测试代码,以下不重复贴,参照此代码进行测试 */ @Test public void testClassSize(){ Person person = new Person(); System.out.println(ClassLayout.parseInstance(person).toPrintable()); }
打印结果(默认开启压缩):
关闭压缩(ide添加参数:-XX:-UseCompressedOops)
idea修改参数:
添加参数用空格隔开
- 示例2:
public class Person { private int id; private String name; private int age; private LocalDate birthday; }
开启压缩普通对象指针时,对象大小:(8+4)+(4+4+4+4)+4=32字节,补齐了4字节。
关闭压缩普通对象指针时,对象大小:(8+8)+(4+8+4+8)+0=40字节,补齐了0字节。
测试代码同上,不重复贴
默认压缩:
关闭压缩:
- 示例3:
public class Person { static int[] nums = {1, 2, 3}; }
开启压缩普通对象指针时,对象大小:(8+8)+12+4=32字节,补齐了4字节。
关闭压缩普通对象指针时,对象大小:(8+8+4+4)+12+4=40字节,补齐了8字节(对象头填充4字节:数组对象会有对象头对象填充;nums对象填充4字节)。
测试代码:
@Test public void testClassSize(){ Person person = new Person(); // 调用数组对象:person.nums System.out.println(ClassLayout.parseInstance(person.nums).toPrintable()); }
默认压缩:
关闭压缩: