目录
在编写Java代码时,我们很少会关注一个Java对象到底占用多少内存,这就可能导致大量内存在无形中被浪费掉了。
一个Java对象到底有多大?
想要精确计算一个Java对象占用的内存,我们首先要了解Java对象的结构表示。
一个Java对象在Heap的表示,可以分为三部分:
-
Object Header
-
Class Pointer
-
Fields
1.Object Header
头信息(Object Header)是必不可少的,它记录着对象的状态。
其在32位与64位占用空间不同,在32位中:
64位中:
2.Class Pointer
在Java中,一切皆对象。每个类都有一个父类,Class Pointer就是当前对象父类的一个指针,在32位系统中,这个指针为4byte;在64位系统中,如果开启指针压缩(-XX:+UseCompressedOops)或者JVM堆的最大值小于32G,这个指针也是4byte,否则是8byte。
3.Fields
关于字段(Fields),这里指的是类的实例字段;也就是说不包括静态字段,因为这个字段是共享内存的,只会存在一份。
下面以32位系统为例子,计算一下java.lang.Integer到底占用多大内存:Object Header 和 Pointer 都是固定的,4+4=8byte;再看看字段,只有这一个,表示数值:
/**
* The value of the <code>Integer</code>.
*
* @serial
*/
private final int value;
一个int在java中占据4byte,所以Integer的大小为4+4+4=12byte。
还需要提醒额外的一点:Java中,对象占用的heap大小是8位对齐,上面的12byte没有对齐,所以需要补位4byte。结果是16byte!
而考虑到数组这一特殊对象,我们计算数组长度的时候,需要额外加上一个长度的字段,即一个int的大小。
节约内存原则
在简单了解了对象的内存使用情况后,我们可以对内存进行规划。
一个java.lang.Integer占用16byte,而一个int占用4byte,4:1的比例,也就是说整数的类类型是基本类型内存的4倍!由此我们得出第一个节约内存的原则:
(1)尽量使用基本类型,而不是包装类型。
数据库建表的时候各字段类型使用需要仔细推敲,同样JavaBean中的属性字段类型也需要仔细斟酌。不要吝啬使用byte,short,boolean等。如果短类型能放下数据,尽量不要使用更长的类型。虽然一个long只比一个int多4byte,但如果内存中有100W个long,那就白白浪费了约4MB空间,不要小看任何的空间浪费,内存是节省出来的。所以:
(2)斟酌字段类型,在满足容量前提下,尽量用小字段。
一个ArrayList集合,如果里面放了10个数字,占用多少内存吗?让我们算算:
/**
* The array buffer into which the elements of the ArrayList are stored.
* The capacity of the ArrayList is the length of this array buffer.
*/
private transient Object[] elementData;
/**
* The size of the ArrayList (the number of elements it contains).
*
* @serial
*/
private int size;
Object Header占4byte,Pointer占4byte,一个int字段占4byte,elementData数组本身占12byte,数组中10个Integer对象占10×16byte。
所以整个集合空间大小为4+4+4+12+160=184byte。
如果我们用int[]代替集合呢,12+4×10=52byte,对齐后是56byte。
集合跟数组的比例是184:56,已超过3:1!所以我们的第三个建议是:
(3)如果可能,尽量用数组,少用集合。
合理控制内存的使用,提高JVM的使用效率。
欢迎关注我们,做技术,我们是认真的!