阅读本文大概需要 2.8 分钟。

出处:http://u6.gg/swLPg

编写 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 中,一切皆对象;每个类都有一个父类, ClassPointer 就是当前对象父类的一个指针。

在 32 位系统中,这个指针为 4byte;

在 64 位系统中,如果开启指针压缩(-XX:+UseCompressedOops)或者 JVM 堆的最大值小于 32G,这个指针也是 4byte,否则是 8byte。

关于字段(Fields),这里指的是类的实例字段;也就是说不包括静态字段,因为这个字段是共享内存的,只会存在一份。

下面以 32 位系统为例子,计算一下 java.lang.Integer 到底占用多大内存:

ObjectHeader 和 Pointer 都是固定的,4+4=8byte 。

再看看字段,只有这一个,表示数值:

/** * The value of the Integer. * * @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 集合中绝大部分的实现,而且比较省内存。

小技巧

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

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

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

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

总结

性能和可读性向来就有些矛盾,在这里也是,为了节约内存,不得不进行取舍,代码丑陋了一些,可读性差了一些,还好能省下一些内存。

上面的原则在确实需要节约内存的时候 ,不妨可以试试!

·END·

程序员的成长之路

路虽远,行则必至

本文原发于 同名微信公众号「程序员的成长之路」,回复「1024」你懂得,给个赞呗。

回复 [ 520 ] 领取程序员最佳学习方式

回复 [ 256 ] 查看 Java 程序员成长规划

往期精彩回顾

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

  1. java string最大长度_一个Java字符串中到底有多少个字符?

    作者:鸟窝 依照Java的文档, Java中的字符内部是以UTF-16编码方式表示的,最小值是 (0),最大值是(65535), 也就是一个字符以2个字节来表示,难道Java最多只能表示 65535个 ...

  2. 一个java类可以有_一个.java文件中可以有几个同级类?

    1.在一个.java文件中可以有多个同级类(和public一样的位置,注意不是内部类).其修饰符只可以public/abstract/final/和无修饰符,不能是其他的protected/priva ...

  3. java背单词软件_一个JAVA写的背单词程序

    一个JAVA写的背单词程序 2007-6-9文字大小:大中小 俺看了一些Java, 写个程序出来玩玩.由于界面是用Jbuilder生成的,可能代码比较乱,而且还没合起来. 目前版本是0.00001 / ...

  4. java命令行参数_一个 java 命令行参数顺序的坑

    tream Stream是在Java SE 8 API添加的用于增强集合的操作接口,可以让你以一种声明的方式处理集合数据.将要处理的集合看作一种流的创建者,将集合内部的元素转换为流并且在管道中传输, ...

  5. java恶作剧小程序_一个Java恶搞小程序

    运用Java程序控制某个应用程序的运行(以网易云音乐为例),步骤如下 1.建立bat文件分别是start.bat(控制程序的运行)和kill.bat(控制程序的结束): start.bat 的内容如下 ...

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

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

  7. 第一个java程序的错误_我是一名java初学者,执行第一个java程序welcome.java出现了以下错误,这是为什么?...

    我是一名java初学者,执行第一个java程序welcome.java出现了以下错误,这是为什么? welcome.java: import javax.swing.*; public class w ...

  8. 达内java面试题集_达内java面试题

    JAVA面试题-COREJAVA部分1.在main(String[] args)方法内是否可以调用一个非静态方法?答案:不能2.同一个文件里是否可以有两个public类?答案:不能3.方法名是否可以与 ...

  9. java项目----教务管理系统_基于Java的教务管理系统

    java项目----教务管理系统_基于Java的教务管理系统 2022-04-22 18:18·java基础 最近为客户开发了一套学校用教务管理系统,主要实现学生.课程.老师.选课等相关的信息化管理功 ...

最新文章

  1. vs 编译android so
  2. 零基础python必背代码-零基础小白Python入门必看:通俗易懂,搞定深浅拷贝
  3. live555 源码分析:RTSPServer 组件结构
  4. 工业机器人国内外的发展现状
  5. python学习之路-day8
  6. Rust 语言本身的问题
  7. 谷歌浏览器chrome安装Hackbar插件方式
  8. 单机实验mysql的主从备份(master slaver)
  9. CCF为何能吸引6.8万付费会员?
  10. Python基础知识之二
  11. C语言———求”完数“
  12. 大数据技术在电子政务中的应用
  13. 执行oracle 函数,oracle 函数function语法及简单实例
  14. 正态分布假设检验相关知识
  15. MySQL数据文件的组织
  16. 交直流调速系统matlab,MATLAB与交直流调速系统仿真.ppt
  17. 集合工具类 以及 几个小面试题(是我保存的)
  18. okdownload下载mp4视频文件
  19. WebSocket 协议以及 Socket 接口
  20. 第七节 设计电路与学书法学绘画并没有差别

热门文章

  1. springboot接口返回封装与异常控制
  2. Pyinstaller打包附带DLL、图标和压缩EXE方法
  3. springcloud(八):配置中心服务化和高可用
  4. Java —— Reflect反射机制
  5. python 和scikit安装
  6. Sublime 自定义格式化快捷键
  7. HDU 1754 I Hate It(线段树版)
  8. 为所有北京奥运冠军名字作诗(诗集)
  9. 把应用程序从 Internet Explorer 迁移到 Mozilla
  10. [转载] python中断响应_用Python脚本监测.py脚本的进程状态,并实现中断重启。