本文目录一览:
如何计算java对象占用的内存
Java有一个很好的地方就是java的垃圾收集机制,这个机制集成于jvm的,对程序员来说是隐藏且不透明的。这种情况下,如何得到某个对象消耗的内存呢?
曾经看到过有人用以下方法来计算:在生成该object的前后都调用java.lang.Runtime.freeMemory()方法,然后看两者之差即为该object消耗的内存量。
这种方法的代码是:
long totalMem = java.lang.Runtime.freeMemory();
Object myBigObject = null;
System.out.println("You just got rid of " + totalMem
- java.lang.Runtime.freeMemory());
这种想法是对的,但是实际上,jvm的freememory往往不能正确反应实际的free
memory。比如在jvm要进行垃圾收集的时候,free
memory就会缩小。而如果决定垃圾收集的时间发生在该object生成之后,而在第二次调用java.lang.Runtime.freeMemory()之前,那么就会错误地增加该object消耗的内存量。
在java专家By
Tony Sintes的文章"Discover how much memory an object consumes "
里面提到了应该用Runtime.getRuntime().totalMemory();并且计算两次之差来得到消耗的内存量。
By Tony Sintes的源代码:
public class Memory {
private final static int _SIZE = 500;
public static void main( String [] args )
throws Exception {
Object[] array = new Object[_SIZE];
Runtime.getRuntime().gc();
long start = Runtime.getRuntime().totalMemory();
for (int i = 0; i _SIZE; i++) {
array[i] = new Object();
}
Runtime.getRuntime().gc();
long end = Runtime.getRuntime().totalMemory();
long difference = ( start - end ) / _SIZE;
System.out.println( difference + " bytes used
per object on average" );
}
}
实际上,这种方法基本上正确了,但是By Tony Sintes疏忽了一点,就是仅仅Runtime.getRuntime().gc();并不能真正完成垃圾收集,也就是说实际上jvm的内存此时并不是稳定的。
所以,只有当内存不再发生大的变动,或者说已经稳定,我们才可能说垃圾收集已经完成。
如何才能真正确保基本完成了jvm的垃圾收集呢?实现这个功能的代码如下:
private static final Runtime s_runtime =
Runtime.getRuntime ();
private static long usedMemory ()
{
return s_runtime.totalMemory () -
s_runtime.freeMemory ();
}
private static void runGC () throws Exception
{
long usedMem1 = usedMemory (), usedMem2 = Long.MAX_value;
for (int i = 0; (usedMem1 usedMem2) (i 500); ++ i)
{
s_runtime.runFinalization ();
s_runtime.gc ();
Thread.currentThread ().yield ();
usedMem2 = usedMem1;
usedMem1 = usedMemory ();
}
}
一个Java对象到底占多大内存
对象头
对象头在32位系统上占用8bytes,64位系统上占用16bytes。
实例数据
原生类型(primitive type)的内存占用如下:
Primitive Type Memory Required(bytes)
boolean 1
byte 1
short 2
char 2
int 4
float 4
long 8
double 8
reference类型在32位系统上每个占用4bytes, 在64位系统上每个占用8bytes。
对齐填充
HotSpot的对齐方式为8字节对齐:
(对象头 + 实例数据 + padding) % 8等于0且0 = padding 8
指针压缩
对象占用的内存大小收到VM参数UseCompressedOops的影响。
1)对对象头的影响
开启(-XX:+UseCompressedOops)对象头大小为12bytes(64位机器)。
static class A {
int a;
}
一个Java对象到底占用多大内存
abstractclassSizeOf{privatefinalRuntimes_runtime=Runtime.getRuntime();/****子类负责覆盖该方法以提供被测试类的实例**@return被测试类的实例*/protectedabstractObjectnewInstance();/****计算实例的大小(字节数)**@return实例所占内存的字节数*@throwsException*/publicintsize()throwsException{//垃圾回收runGC();//提供尽可能多(10万)的实例以使计算结果更精确finalintcount=100000;Object[]objects=newObject[count];//实例化前堆已使用大小longheap1=usedMemory();//多实例化一个对象for(inti=-1;i=0){objects[i]=object;}else{//释放第一个对象object=null;//垃圾收集runGC();//实例化之前堆已使用大小heap1=usedMemory();}}runGC();//实例化之后堆已使用大小longheap2=usedMemory();finalintsize=Math.round(((float)(heap2-heap1))/count);//释放内存for(inti=0;icount;++i){objects[i]=null;}objects=null;returnsize;}privatevoidrunGC()throwsException{//执行多次以使内存收集更有效for(intr=0;r4;++r){_runGC();}}
如何计算Java对象所占内存的大小
java中可以用.getBytes().length获取字符串占用内容的大小,原理是java中任何字符都采用Unicode编码,所以衡量占用内存大小采用占用的字节数。
举例如下:
public class TestStringSize {
public static final void main(String[] args) {
System.out.println("占用内存大小:"+"学java".getBytes().length);
}
}
输出结果:
占用内存大小:6 byte