我们在Java单个对象内存布局中讲解了单个简单的Java对象所占内存的大小的计算。那么这篇文章主要是讲解复杂Java对象所占内存大小的计算,我们把继承、复合的对象称为复杂对象

继承对象

class Parent {
protected int x; // 4字节
protected int y; // 4字节 protected boolean flag; // 1字节
} class Child extends Parent {
private int z; // 4字节
} public class ExtendsObjectSizer {
public static void main(String[] args) {
System.out.println("继承对象的大小为:" + ObjectSizeFetcher.sizeOf(new Child()) + "字节");
}
}

  

然后重新打包,执行如下的命令:

## 没有开启指针压缩功能
java -XX:-UseCompressedOops -javaagent:ObjectSizeFetcherAgent-1.0-SNAPSHOT.jar com.twq.ExtendsObjectSizer

得到的结果如下:

可以看出new Child()的内存大小为40字节,那么这个40字节是怎么来的呢?我们看下图:

40字节 = 16 + (4 + 4 + 1 + 7) + 4 + 7

## 开启指针压缩功能
java -XX:+UseCompressedOops -javaagent:ObjectSizeFetcherAgent-1.0-SNAPSHOT.jar com.twq.ExtendsObjectSizer

  得到的结果如下:

可以看出new Child()的内存大小为32字节,那么这个32字节是怎么来的呢?我们看下图:

32字节 = 12 + 4 + 4 + 1 + 7 + 4

总结:继承对象也只有一个对象头,继承对象所占内存的大小由4部分组成:

  1. 对象头
  2. 父亲属性所占内存大小,需要对齐补充至8的倍数
  3. 子类属性所占内存大小
  4. 对齐补充

复合对象

我们接下来看下复合对象所占内存大小,先看下如下代码:

class Employee {
private int age; // 4字节
private double high; // 8字节
} class Dept {
private int num; // 4字节 private Employee[] employees = new Employee[3]; public Dept() {
for (int i = 0; i < employees.length; i++) {
employees[i] = new Employee();
}
}
} public class CompositeObjectSizer {
public static void main(String[] args) throws IllegalAccessException {
System.out.println("复合对象内存的大小为:" + ObjectSizeFetcher.sizeOf(new Dept()) + "字节");
System.out.println("复合对象内存的总大小为:" + ObjectSizeFetcher.fullSizeOf(new Dept()) + "字节");
}
}

  然后重新打包,执行如下的命令:

得到的结果如下:

可以看出复合对象内存的大小为32字节 = 对象头16字节 + num属性4字节 + employees引用类型的大小8字节 + 对齐补充4字节

这个32字节也只是直接计算当前Dept对象占用空间大小,这个大小并没有包含数组employees中所有的Employee的内存大小,那么这个复合对象的总大小为176字节,这个大小包括了数组employees中所有的Employee的内存大小,也就是递归计算当前对象占用空间总大小。

递归计算复合对象占用的内存的时候需要注意的是:对齐填充是以每个对象为单位进行的,看下面这个图就很容易明白: 

根据上图,我们手动来计算一下new Dept()的总内存大小,分为3部分:

  • Dept实例对象本身的大小为:16 + 4 + 8 + 4 = 32字节
  • 数组对象employees的大小为:24 + 8 * 3 = 48字节
  • 3个Employee对象的大小为:3 * (16 + 4 + 8 + 4) = 96字节

所以内存总大小为:32 + 48 + 96 = 176字节

我们再来看下开启了指针压缩功能的复合对象内存的大小情况,我们执行如下的命令:

## 开启指针压缩功能
java -XX:+UseCompressedOops -javaagent:ObjectSizeFetcherAgent-1.0-SNAPSHOT.jar com.twq.CompositeObjectSizer

  

得到的结果如下:

可以看出复合对象内存的大小为24字节 = 对象头12字节 + num属性4字节 + employees引用类型的大小4字节 + 对齐补充4字节

在开启指针压缩功能的情况下,Dept复合对象的总大小为128字节,如下图:

根据上图,我们手动来计算一下new Dept()的总内存大小,分为3部分:

  • Dept实例对象本身的大小为:12 + 4 + 4 + 4 = 24字节
  • 数组对象employees的大小为:16 + 4 * 3 + 4 = 32字节
  • 3个Employee对象的大小为:3 * (12 + 4 + 8) = 72字节

所以内存总大小为:24 + 32 + 72 = 126字节

ArrayList

以下是ArrayList的部分源码:

public abstract class AbstractList<E> extends AbstractCollection<E> implements List<E> {
protected transient int modCount = 0;
} public class ArrayList<E> extends AbstractList<E>
implements List<E>, RandomAccess, Cloneable, java.io.Serializable
{
private static final long serialVersionUID = 8683452581122892189L; /**
* Default initial capacity.
*/
private static final int DEFAULT_CAPACITY = 10; /**
* Shared empty array instance used for empty instances.
*/
private static final Object[] EMPTY_ELEMENTDATA = {}; private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {}; transient Object[] elementData; // non-private to simplify nested class access /**
* The size of the ArrayList (the number of elements it contains).
*/
private int size;
}

  

static变量属于类,不属于实例,存放在全局数据段。普通变量才纳入Java对象占用空间的计算,一个用于存放数组元素的Object[], 一个int类型的size,还有一个是父类中int类型的modCount。因此:

  • 在64位操作系统,且未开启指针压缩功能的前提下,new ArrayList<Integer>()的内存大小应为:对象头16字节 + 父类属性modCount大小4字节 + 对齐补充4字节 + 子类Object[]引用类型的8字节 + 子类int类型的属性大小4字节 + 对齐补充4字节 = 40字节
  • 在64位操作系统,且开启了指针压缩功能的前提下,new ArrayList<Integer>()的内存大小应为:对象头12字节 + 父类属性modCount大小4字节 + 对齐补充4字节 + 子类Object[]引用类型的4字节 + 子类int类型的属性大小4字节 + 对齐补充4字节 = 32字节

复杂Java对象所占内存的大小的更多相关文章

  1. 如何计算Java对象所占内存的大小

    [ 简单总结: 随便一个java项目,引入jar包: lucene-core-4.0.0.jar 如果是 maven项目,直接用如下依赖: <dependency> <groupId ...

  2. java中基本类型封装对象所占内存的大小(转)

    这是一个程序,java中没有现成的sizeof的实现,原因主要是java中的基本数据类型的大小都是固定的,所以看上去没有必要用sizeof这个关键字. 实现的想法是这样的:java.lang.Runt ...

  3. 4种方法教你如何查看java对象所占内存大小

    摘要:本文讲述4种查看java对象所占内存大小的方法 本文分享自华为云社区<查看java对象所占内存大小>,作者:xiewenci. 计算java对象所占内存大小 1.使用jdk8自带AP ...

  4. 如何获取一个Java对象所占内存大小

    新建一个maven工程 我们先在IDEA中新建一个名为ObjectSizeFetcherAgent的maven工程,如下图: 在maven项目中的pom.xml中新增一个打jar包的插件,如下: &l ...

  5. Java学习日记-2.3 基本数据类型和对象所占内存空间大小

    转自:http://www.newsmth.net/nForum/#!article/Java/324167

  6. JAVA数组所占内存大小的对比

    1.两个数据模型 第一个是基本类型数组,第二个使用的是Float对象数组 public class SummaryModel{ private float[] summaryData; public ...

  7. C# 能否获取一个对象所占内存的大小?

    今日,在项目重构的时候忽然想到一个问题,一个类哪些成员的增加,会影响一个类所占内存的大小?C#有没有办法知道一个对象占多少内存呢? 第一个问题:很快想到是类的非静态的字段.属性. 第二个问题:首先想到 ...

  8. 获取JAVA对象占用的内存大小

    介绍两种获取JAVA对象内存大小的方法. 第一种:Instrumentation 简介: 使用java.lang.instrument 的Instrumentation来获取一个对象的内存大小.利用I ...

  9. 一个Java对象到底占多大内存

    最近在读<深入理解Java虚拟机>,对Java对象的内存布局有了进一步的认识,于是脑子里自然而然就有一个很普通的问题,就是一个Java对象到底占用多大内存? 在网上搜到了一篇博客讲的非常好 ...

随机推荐

  1. python局部变量和全局变量(6)

    在python开发中,变量也是有生命周期的,一旦周期结束,程序会自动清理暂用的空间,释放内存,变量分为两者,一种是局部变量,一种是全局变量,两者具体有什么区别呢…… 一.局部变量 一般而言在函数内部或 ...

  2. 2、Maven的简介和配置

    1.下载Maven  apche-maven-3.5.2 2. 三.maven简介 1.基于Ant的构建工具,Ant有的功能Maven都有,额外添加了其他的功能 2.运行原理图 2.1本地仓库:计算机 ...

  3. LeetCode 75. 颜色分类(Sort Colors) 30

    75. 颜色分类 75. Sort Colors 题目描述 给定一个包含红色.白色和蓝色,一共 n 个元素的数组,原地对它们进行排序,使得相同颜色的元素相邻,并按照红色.白色.蓝色顺序排列. 此题中, ...

  4. git stash详解

        应用场景: 1 当正在dev分支上开发某个项目,这时项目中出现一个bug,需要紧急修复,但是正在开发的内容只是完成一半,还不想提交,这时可以用git stash命令将修改的内容保存至堆栈区,然 ...

  5. springboot注入的四个注解

    java配置主要靠java类和一些注解来达到和xml配置一样的效果,比较常用的注解有: @Configuration:声明一个类作为配置类,代替xml文件@Bean:声明在方法上,将方法的返回值加入B ...

  6. 通过命令窗口导入导出oracle数据库到dmp文件

    通过命令窗口导入导出oracle数据库到dmp文件 很多时候我们需要备份Oracle的数据库,然后将数据导入其他数据库,因为有大文本字段会导致insert无法完全导出,只能导出为dmp文件,前提是wi ...

  7. python多线程爬取世纪佳缘女生资料并简单数据分析

    一. 目标 ​ 作为一只万年单身狗,一直很好奇女生找对象的时候都在想啥呢,这事也不好意思直接问身边的女生,不然别人还以为你要跟她表白啥的,况且工科出身的自己本来接触的女生就少,即使是挨个问遍,样本量也 ...

  8. 聊聊 ES6 中的箭头函数

    首先来两点: 当只有一个参数的时候,那么 () 可以省略 当只有一个 return 的时候,那么 {} 可以省略 当函数体内只有一条语句的时候,那么 {} 也可以省略 下面来几个简单的例子来对比 ES ...

  9. 【亲测可行,图片宽度高度自适应】c# Graphics MeasureString精确测量字体宽度

    , , ) { int count = number.Length; //需要配置的字段 //Font f = new Font("Microsoft Sans Serif", f ...

  10. 如何通过Restful API的方式读取SAP Commerce Cloud的Product图片

    需求:我在SAP Commerce Cloud的backoffice里给某product维护了一些图片: 分别位于Normal,Thumbnails和Others等字段: 现在我想通过Restful ...