一个java对象占多大内存呢
做java研发的可能平时多少都会接触到这个问题,但是就业务开发来说只要不是追求性能极致,一般不会对这个去问题深入探究,我们熟知的基础类型大家可能都知道 byte(1字节) short(2字节) int(4字节) long(8字节)等,那么一个对象在内存中到底占多大内存呢?
废话不多说,我们先看下面的几个例子。
public class A {}
public class B {
private long s;
}
public class C {
private int a;
private long s;
}
int[] aa = new int[0];
主函数:
public class Test {
public static void main(String[] args) {
A a = new A();
System.out.println(ClassLayout.parseInstance(a).toPrintable());
B b = new B();
System.out.println(ClassLayout.parseInstance(b).toPrintable());
C c = new C();
System.out.println(ClassLayout.parseInstance(c).toPrintable());
int[] aa = new int[0];
System.out.println(ClassLayout.parseInstance(aa).toPrintable());
}
}
注明:1.运行坏境为64位虚拟机。
2.主函数里所用到的包为jol工具,引入方式如下。
<dependency> <groupId>org.openjdk.jol</groupId> <artifactId>jol-core</artifactId> <version>0.9</version> </dependency>
输出结果如下:
结果如上图所示,A对象为16bytes,B对象为24bytes,C对象为24bytes,数组对象为16bytes。
java对象的组成部分
1.对象头
在jvm虚拟机中每一个java对象都有一个对象头,对象头中包含标记字段以及对象指针,标记字段用来储存hash码、gc信息以及锁信息,而指针则指向改对象的类。在64位jvm虚拟机中这两部分都是64位的,所以也就是需要128位大小(16 bytes)。
注意:64位虚拟机中在堆内存小于32GB的情况下,UseCompressedOops是默认开启的,该参数表示开启指针压缩,会将原来64位的指针压缩为32位。
2.实例数据
类中所有的实例字段数据。
3.内存填充部分(alignment)
该部分作用是用来保证java对象在虚拟机中占内存大小为8N bytes。
4.数组长度
这个是数组对象才特有的。
java的原生数据类型所占内存情况如下表格:
boolean | byte | short | char | int | long | float | double |
1 bytes | 1 bytes | 2 bytes | 2 bytes | 4 bytes | 8 bytes | 4 bytes | 8 bytes |
接下来我们来挨个分析。
A对象,A对象就是一个空对象,所以在内存中占据大小就是对象头的大小等于16 bytes,由于开启指针压缩则对象头占据大小为12 bytes,但是12 bytes字节不是8的倍数所以需要进行内存对齐,最后加上了4 bytes的空白字节,最终该对象占据16 bytes大小
B对象,B对象包含一个long基本数据类型,所以大小等于 12 bytes+8 bytes=20 bytes,20 bytes不是8N所以需要加上4 bytes的填充字节,最终该对象占据24 bytes。
C对象,C对象包含一个long、一个int基本数据类型,所以大小等于 12 bytes+8 bytes+4 bytes=24 bytes,正好8N,不需要内存填充,最终该对象占据24 bytes。
D对象,D对象是一个数组对象,因为数组对象会多一个数组长度存储部分,所以大小等于12 bytes+4 bytes=16 bytes,注意这里数组长度我给的是0,如果给的是1大家可以试试看输出结果又是多少呢。
至此我们已经分析完毕,感兴趣的小伙伴可以去试一下,加深下对对象内存大小的理解,或者更改下虚拟机的运行参数UseCompressedOops 设置为-XX:-UseCompressedOops关闭指针压缩看看结果又是多少。