Object
数组
JVM中数组也是对象,但是和普通对象不同的是数组的对象头中,还有4字节是用来表示数组大小的。
注意最后面的Object数组,如果数组中存储的不是基础类型,那么实际上存储的是执行该对象的指针,该指针大小是4个字节。
String
String是一个非常特殊的对象,它的底层是以byte数组存储的。
注意,在JDK9之前,String的底层存储结构是char[],一个char需要占用两个字节的存储单位。
因为大部分的String都是以ISO字符编码来表示的,只需要一个字节存储就够了,两个字节完全是浪费。
于是在JDK9之后,字符串的底层存储变成了byte[]。
可以看到String中的对象头是12字节,再加上hash,coder,和hasIsZero属性,以及对齐的两字节,最后加上4字节的指针指向一个byte数组,最后的大小是24字节。
ArrayList
我们构建一个空的ArrayList:
最后
可以看到正如前面所说,Hotspot VM中,自动内存管理系统要求对象的起始地址必须是8字节的整数倍,换句话说就是任何对象的大小都必须为8字节的整数倍,对象头部分的大小已经精心设计为8字节的1倍或2倍。因此如果对象的实例数据部分没有对齐的话,则需要使用对其填充。
代码
public class ObjectDemo {
public static void objectSize() {
int[] arrary1 = new int[100];
Arrays.fill(arrary1, 10);
double[] arrary2 = new double[100];
Arrays.fill(arrary2, 10);
String[] arrary3 = new String[100];
Arrays.fill(arrary3, "test");
String s = new String("");
List<Integer> list = new ArrayList<>();
Object object = new Object();
System.out.println();
System.out.println(ClassLayout.parseInstance(arrary1).toPrintable());
System.out.println();
System.out.println(ClassLayout.parseInstance(arrary2).toPrintable());
System.out.println();
System.out.println(ClassLayout.parseInstance(arrary3).toPrintable());
System.out.println();
System.out.println(ClassLayout.parseInstance(s).toPrintable());
System.out.println();
System.out.println(ClassLayout.parseInstance(list).toPrintable());
System.out.println();
System.out.println(ClassLayout.parseInstance(object).toPrintable());
}
public static void main(String[] args) {
// TODO Auto-generated method stub
objectSize();
}
}