JOL(Java Object Layout)是分析JVM中对象内存布局的工具。
注:本文基于JDK 1.8分析
查看对象布局
- 对象头占用的空间大小;
- 对象中的字段占用的空间;
- 为了字节对齐而产生的额外数据。
public static void main(String[] args) {
//当前虚拟机信息
System.out.println(VM.current().details());
//A类的内存布局
System.out.println(ClassLayout.parseClass(A.class).toPrintable());
}
OFF 是每个部分的开始字节偏移量,SZ 是该部分的占用的字节数,TYPE 是类型,DESCRIPTION 是每个部分的描述信息,VALUE 是每个部分的值。
对象够包括两部分 mark 和 class,分别占用 8 和 4 字节。接下是 A 类中的字段 f 占用1个字节。最后的3个字节是为了字节对齐填充的3个字节。实例的大小为16字节。最后的 Space losses 指的是空间浪费,该对象最后填充了3个字节,即浪费的内存空间为3个字节。
对齐
因为底层硬件平台会要求访问时对齐,从而维护访问性能和正确性,这就希望字段按照大小对齐。对于布尔类型没什么影响,但是对于 long 类型的就比较困难。在本例中,可以看到 long `字段确实按照8个字节对齐了,但是有时会在对象头之后产生空隙。
public static void main(String[] args) {
System.out.println(ClassLayout.parseClass(A.class).toPrintable());
}
class A {
long f;
}
4个字节就是为了补齐对象头产生的空隙。而long 类型字段占用了8个字节。对象总共占用24个字节。从 Space losses 可以看到与第一个例子不同,这里的空间浪费是在对象内部(internal),也就是为了补齐对象头而使用的4个字节。
字段重排
public static void main(String[] args) {
System.out.println(ClassLayout.parseClass(A.class).toPrintable());
}
class A {
boolean bo1, bo2;
byte b1, b2;
char c1, c2;
double d1, d2;
float f1, f2;
int i1, i2;
long l1, l2;
short s1, s2;
}
可以看到内存中的字段顺序与类中声明的完全不同。在对象头之后先用4字节的 float 类型变量 f1 进行填充。之后就按照先大后小的顺序排列字段,也就是 8->4->2->1 这样的顺序来存储字段。
继承中的字段布局
继承关系中,JVM会首先存放超类的字段。
public static void main(String[] args) {
System.out.println(ClassLayout.parseClass(C.class).toPrintable());
}
class A {
int a;
}
class B extends A {
int b;
}
class C extends B {
int c;
}
超类中内存布局的缺陷
在继承关系中的内存布局,在超类之前对齐间隙。
public static void main(String[] args) {
System.out.println(ClassLayout.parseClass(C.class).toPrintable());
}
class A {
long a;
}
class B extends A {
long b;
}
class C extends B {
int c;
long d;
}
继承产生的间隙
内存中并没有按照预想的方式来存放
public static void main(String[] args) {
System.out.println(ClassLayout.parseClass(C.class).toPrintable());
}
class A {
boolean a;
}
class B extends A {
boolean b;
}
class C extends B {
boolean c;
}