如何计算Java对象占用了多少空间?

在Java中没有sizeof运算符,所以没办法知道一个对象到底占用了多大的空间,但是在分配对象的时候会有一些基本的规则,我们根据这些规则大致能判断出来对象大小。

如何计算Java对象占用了多少空间?

对象头

对象的头部至少有两个WORD,如果是数组的话,那么三个WORD,内容如下:

  1. 对象的HashCode,锁信息等

  2. 到对象类型数据的指针

  3. 数组的长度(如果是数组的话)

规则

首先,任何对象都是8字节对齐,属性按照[long,double]、[int,float]、[char,short]、[byte,boolean]、reference的顺序存放,举个例子:


  1. public class Test { 
  2.     byte a; 
  3.     int b; 
  4.     boolean c; 
  5.     long d; 
  6.     Object e; 

如果这个对象的属性按照顺序存放的话,要占用的空间为:head(8) + a(1) + padding(3) + b(4) + c(1) + padding(7) + d(8) + e(4) + padding(4) = 40。但是按照这个规则得到:head(8) + d(8) + b(4) + a(1) + c(1) + padding(2) + e(4) + padding(4) = 32。可以看到节省了不少空间。

在涉及继承关系的时候有一个最基本的规则:首先存放父类中的成员,接着才是子类中的成员,举个例子:


  1. class A { 
  2.     long a; 
  3.     int b; 
  4.     int c; 
  5. class B extends A { 
  6.     long d; 

这样存放的顺序及占用空间如下:head(8) + a(8) + b(4) + c(4) + d(8) = 32。那如果父类中的属性不够八个字节怎么办?这样就有了新的一条规则:父类中最后一个成员与子类的第一个成员的间隔如果不够4个字节,此时需要扩展到4 个字节的基本单位,举个例子:


  1. class A { 
  2.     byte a; 
  3. class B extends A { 
  4.     byte b; 

那么此时占用的空间如下:head(8) + a(1) + padding(3) + b(1) + padding(3) = 16。显然这种方式比较浪费空间,那么就有了:如果子类的第一个成员是double或者long,并且父类并没有用完8个字节,JVM会破坏规将较小的数 据填充到该空间,举个例子:


  1. class A { 
  2.     byte a; 
  3. class B extends A { 
  4.     long b; 
  5.     short c; 
  6.     byte d; 

此时占用的空间如下:head(8) + a(1) + padding(3) + c(2) + d(1) + padding(1) + b(8) = 24。


来源:51CTO

上一篇:SonarQube安装步骤 Windows10系统


下一篇:操作系统——设置虚拟内存(Virtual Memory)的原理