博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
一个Java对象到底占用多大内存
阅读量:6059 次
发布时间:2019-06-20

本文共 2428 字,大约阅读时间需要 8 分钟。

hot3.png

最近对对象的内存布局有了进一步的认识,于是脑子里自然而然就有一个很普通的问题,就是一个对象到底占用多大内存?

  在网上搜到了一篇博客讲的非常好,里面提供的这个类也非常实用:

  View Code

  大家可以用这个代码边看边验证,注意的是,运行这个程序需要通过javaagent注入Instrumentation,具体可以看原博客。我今天主要是总结下手动计算Java对象占用字节数的基本规则,做为基本的技能必须get√,希望能帮到和我一样的Java菜鸟。

  在介绍之前,简单回顾下,Java对象的内存布局:对象头(Header),实例数据(Instance Data)和对齐填充(Padding),详细的可以看我的读书笔记。另外:不同的环境结果可能有差异,我所在的环境是HotSpot虚拟机,64位Windwos。

  下面进入正文:

  对象头对象头在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; }

  A对象占用内存情况:

  关闭指针压缩: 16+4=20不是8的倍数,所以+padding/4=24

  开启指针压缩: 12+4=16已经是8的倍数了,不需要再padding。

  1) 对reference类型的影响

  64位机器上reference类型占用8个字节,开启指针压缩后占用4个字节。

  static class B2 { int b2a; Integer b2b; }

  B2对象占用内存情况:

  关闭指针压缩: 16+4+8=28不是8的倍数,所以+padding/4=32

  开启指针压缩: 12+4+4=20不是8的倍数,所以+padding/4=24

  数组对象64位机器上,数组对象的对象头占用24个字节,启用压缩之后占用16个字节。之所以比普通对象占用内存多是因为需要额外的空间数组的长度。

先考虑下new Integer[0]占用的内存大小,长度为0,即是对象头的大小:

  未开启压缩:24bytes

  开启压缩后:16bytes

  接着计算new Integer[1],new Integer[2],new Integer[3]和new Integer[4]就很容易了:

  未开启压缩:

  开启压缩:

  拿new Integer[3]来具体解释下:

  未开启压缩:24(对象头)+8*3=48,不需要padding;

  开启压缩:16(对象头)+3*4=28,+padding/4=32,其他依次类推。

  自定义类的数组也是一样的,比如:

  static class B3 { int a; Integer b; }

  new B3[3]占用的内存大小:

  未开启压缩:48

  开启压缩后:32

  复合对象计算复合对象占用内存的大小其实就是运用上面几条规则,只是麻烦点。

  1)对象本身的大小

  直接计算当前对象占用空间大小,包括当前类及超类的基本类型实例字段大小、引用类型实例字段引用大小、实例基本类型数组总占用空间、实例引用类型数组引用本身占用空间大小; 但是不包括超类继承下来的和当前类声明的实例引用字段的对象本身的大小、实例引用数组引用的对象本身的大小。

  static class B { int a; int b; } static class C { int ba; B[] as = new B[3]; C() { for (int i = 0; i < as.length; i++) { as[i] = new B(); } } }

  未开启压缩:16(对象头)+4(ba)+8(as引用的大小)+padding/4=32

  开启压缩:12+4+4+padding/4=24

  2)当前对象占用的空间总大小

  递归计算当前对象占用空间总大小,包括当前类和超类的实例字段大小以及实例字段引用对象大小。

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

  现在我们来手动计算下C对象占用的全部内存是多少,主要是三部分构成:C对象本身的大小+数组对象的大小+B对象的大小。

  未开启压缩:

  (16 + 4 + 8) + (24+ 8*3) +(16+8)*3 = 152bytes

  开启压缩:

  (12 + 4 + 4) + (16 + 4*3 +4(数组对象padding)) + (12+8+4(B对象padding))*3 = 96bytes

  大家有兴趣的可以试试。

  实际工作中真正需要手动计算对象大小的场景应该很少,但是个人觉得做为基础知识每个Java开发人员都应该了解,另外:对自己写的代码大概占用多少内存,内存中是怎么布局的应该有一个直觉性的认识。

转载于:https://my.oschina.net/weichou/blog/547677

你可能感兴趣的文章
网络空间同样需要“天朗气清”
查看>>
Amy Schumer名列2016年迈克菲最危险名人榜榜首
查看>>
联想创投大数据基于开源技术 助力全球智能制造
查看>>
陌陌6.0核心诉求:改变看脸社交
查看>>
Palo Alto Networks与国际刑警组织签署首个数据交换协议
查看>>
大数据金融渐行渐进
查看>>
Windows 10如何重置本地组策略设置
查看>>
Python并发编程之线程池/进程池
查看>>
基于uiwebview的富文本编辑器实践
查看>>
商用WiFi如何在三四线城市挣到钱?
查看>>
如何设计一个复杂的分布式爬虫系统?
查看>>
云清联盟发布全网协同防护方案 2017将继续“纳新”共赢未来
查看>>
物联网时代如何管理上百万设备?找风河DLM就对了!
查看>>
Hadoop MapReduce如何进行WordCount自主编译运行
查看>>
这三种思维作为前端工程师应该具备
查看>>
构建更加高效的数据中心
查看>>
电商社交数据在大数据风控的应用实践
查看>>
Java深入学习系列之值传递Or引用传递?
查看>>
PaperWeekly 第十三期--最新文章解读
查看>>
石墨文档宣布获数千万元Pre-A轮融资 上线企业版
查看>>