点击上方蓝色“程序猿DD”,选择“设为星标”

回复“资源”获取独家整理的学习资料!

作者 | 李小武

来源 | http://blog.lichengwu.cn/

编写Java代码的时候,大多数情况下,我们很少关注一个Java对象究竟有多大(占据多少内存),更多的是关注业务与逻辑。但是殊不知,在我们不经意间,大量的内存被无形地浪费了。

一个Java对象到底有多大?

想要精确计算一个Java对象占用的内存,首先要了解Java对象的结构表示。

Java对象结构

一个Java对象在Heap的表示,可以分为三部分:

  • Object Header

  • Class Pointer

  • Fields

每个普通Java对象在堆(heap)中都有一个头信息(object header),头信息是必不可少的,记录着对象的状态。

32位与64位占用空间不同,在32位中:

hash(25)+age(4)+lock(3)=32bit

64位中:

unused(25+1)+hash(31)+age(4)+lock(3)=64bit

我们知道,在Java中,一切皆对象;每个类都有一个父类,Class Pointer就是当前对象父类的一个指针,在32位系统中,这个指针为4byte;在64位系统中,如果开启指针压缩(-XX:+UseCompressedOops)或者JVM堆的最大值小于32G,这个指针也是4byte,否则是8byte。

关于字段(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!

另外,在Java中还有一种特殊的对象,数组!没错,这个对象有点特殊,它比其他对象多了一个属性:长度(length)。所以我们计算数组长度的时候,需要额外加上一个长度的字段,即一个int的大小。

例如:int[] arr = new int[10];

arr的占用heap大小为:

4(object header)+4(pointer)+4(length)+4*10(10个int大小)=52byte 由于需要8位对齐,所以最终大小为`56byte`。

节约内存原则

在了解了对象的内存使用情况后,我们可以简单算一笔帐。一个java.lang.Integer占用16byte,而一个int占用4byte,4:1的比例!也就是说整数的类类型是基本类型内存的4倍!由此我们得出第一个节约内存的原则:

(1)尽量使用基本类型,而不是包装类型。

数据库建表的时候字段类型需要仔细推敲,同样JavaBean中的属性字段类型也需要仔细斟酌。不要吝啬使用short,byte,boolean,如果短类型能放下数据,尽量不要使用更长的类型。一个long比一个int才多4byte,但是你要想,如果内存中有100W个long,那就白白浪费了约4MB空间,不要小看这一点点的空间浪费,因为随便一个跑着在线应用的JVM中,对象都能达到上千万!内存是节省出来的。所以:

(2)斟酌字段类型,在满足容量前提下,尽量用小字段。

你知道一个ArrayList集合,如果里面放了10个数字,占用多少内存吗?让我们算算:

ArrayList中有两个字段:

/*** 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字段(size)占4byte,elementData数组本身占12(4+4+4),数组中10个Integer对象占10×16。所以整个集合空间大小为4+4+4+12+160=184byte。

如果我们用int[]代替集合呢,12+4×10=52byte,对其后56byte。

集合跟数组的比例是184:56,超过3:1了!

所以我们的第三个建议是:

(3)如果可能,尽量用数组,少用集合。

数组中是可以使用基本类型的,但是集合中只能放包装类型!

如果实在需要使用集合,推荐一个比较节约内存的集合工具,fastutil。这里面包含了JKD集合中绝大部分的实现,而且比较省内存。

(4)小技巧

在上面的三个原则基础上,提供两个小技巧。

  • 时间用long/int表示,不用Date或者String。

  • 短字符串如果能穷举或者转换成ascii表示,可以用long或者int表示。

小技巧跟具体的场景是数据有关系,可以根据实际情况进行激进优化节省内存。

总结

性能和可读性向来就有些矛盾,在这里也是,为了节约内存,不得不进行取舍,代码丑陋了一些,可读性差了一些,还好能省下一些内存。上面的原则在确实需要节约内存的时候,不妨可以试试!

DD自研的沪牌代拍业务,点击直达

【往期推荐】

超卖 100 瓶茅台的事故分析

2020-12-15

重磅!GitHub 全部源代码被泄露?

2020-12-15

这本空降京东当当新书榜TOP1的“算法小抄”是什么来头?

2020-12-14

Redis 的 8 大数据类型,写得非常好!

2020-12-14

Spring Boot 的2020最后一击:2.4.1、2.3.7、2.2.12 发布

2020-12-13

排名前 16 的 Java 工具类,哪个你没用过?

2020-12-13

扫一扫,关注我

一起学习,一起进步

每周赠书,福利不断

深度内容

推荐加入

欢迎加入知识星球,一起探讨技术架构,交流技术人生。

加入方式,长按下方二维码:

已在知识星球更新如下:

素质二连,走一个

一个 Java 对象到底有多大?相关推荐

  1. 一个Java对象到底有多大?

    点击上方"方志朋",选择"置顶公众号" 技术文章第一时间送达! 出处:http://u6.gg/swLPg 编写Java代码的时候,大多数情况下,我们很少关注一 ...

  2. 一个Java对象到底占用多大内存?

    最近在调研MAT和VisualVM源码实现,遇到一个可疑问题,两者计算出来的对象大小不一致,才有了这样疑惑. 一个Java对象到底占用多大内存? 为了复现这个问题,准备了4个最简单类: class A ...

  3. java 如何循环执行一个对象_一个Java对象到底有多大?

    编写Java代码的时候,大多数情况下,我们很少关注一个Java对象究竟有多大(占据多少内存),更多的是关注业务与逻辑.但是殊不知,在我们不经意间,大量的内存被无形地浪费了. 一个Java对象到底有多大 ...

  4. 一个Java对象到底占多大内存?(转)

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

  5. java有几大对象_一个 Java 对象到底有多大?

    阅读本文大概需要 2.8 分钟. 出处:http://u6.gg/swLPg 编写 Java 代码的时候,大多数情况下,我们很少关注一个 Java 对象究竟有多大(占据多少内存),更多的是关注业务与逻 ...

  6. java工具多,[转帖]一个 Java 工具到底有多大?

    一个 Java 工具到底有多大? https://www.jianshu.com/p/5ad8b16a8f94 出处:http://u6.gg/swLPg 编写 Java 代码的时刻,大多数情况下,我 ...

  7. java占用内存多大_[转帖]一个Java工具到底占用多大内存?

    一个Java工具到底占用多大内存? https://www.jianshu.com/p/194b745884a5 最近在调研MAT和VisualVM源码实现,遇到一个可疑问题,两者盘算出来的工具巨细不 ...

  8. 一个Java对象占用多大内存

    ​ 这个问题一般会出现在稍微高端一点的 Java 面试环节.要求面试者不仅对 Java 基础知识熟悉,更重要的是要了解内存模型. Java 对象模型 HotSpot JVM 使用名为 oops (Or ...

  9. 【趣话编程】一个Java对象的回忆录:垃圾回收

    简介:趣话编程第三期,今天让我们一起去看看一个Java对象的回忆录:垃圾回收. 原文链接 对象的诞生 "你醒啦!",迷迷糊糊中听到一个声音,我睁开了眼睛,发现一个小伙伴正看着我. ...

最新文章

  1. ganglia 监控mysql_Ganglia监控MySQL
  2. linux centos 7安装 apache php 及mariadb
  3. java使用gridview,网格控件GridView在Android中的使用
  4. 【Java虚拟机】Java虚拟机深度讲解、VisualVM工具、JVM调优
  5. groupby的用法
  6. Failed to compile. ./src/utils/request.js Module not found: Error: Can‘t resolve ‘util-merge‘ in ‘C
  7. GDB调试精粹及使用实例
  8. crtsiii型无砟轨道板_自主知识产权CRTSⅢ型轨道板助力,赣深铁路无砟轨道轨道板灌注第一工作面完成...
  9. 我的angularjs源码学习之旅1——初识angularjs
  10. 银行卡掉了一定要在开户银行补吗?
  11. openmv探索_3_测距以及测量物体大小
  12. 小用lambda表达式,查询数组里大于80的个数
  13. 研华工控机改软PLC使用教程
  14. 抗衡微软,UCDOS失策了,金山系软件完胜,不是WPS
  15. iOS-图片可以自适应屏幕大小吗,应该会失真吧?
  16. 阿里巴巴基础设施挑战与芯片策略
  17. 计算机的删除快捷键,电脑上的删除快捷键是什么?
  18. 【智能手环APP for Android 】01 百度地图展示行动轨迹
  19. [Excel]rank函数实现自动排序
  20. idea高版本设置注入飘红(红色波浪线)

热门文章

  1. textarea去掉回车换行
  2. python3 中递归的最大次数
  3. centos7 yum 安装 python3
  4. python3 str bytes bytearray 互相转换
  5. linux c 实现uname命令
  6. golang 调用库函数错误 cannot refer to unexported name
  7. Libgcrypt实现AES加密
  8. wireshark过虑规则
  9. VS2012编译调试WDM驱动(KdPrint无调试信息 debugview win7无调试信息)
  10. 深入理解 C 指针阅读笔记 -- 第二章