java各种类型对象占用内存情况分析
作者:狮少 发布时间:2023-08-22 10:32:05
前言
其实一般的程序猿根本不用了解这么深,只有当你到了一定层次,需要了解jvm内部运行机制,或者高并发多线程下,你写的代码对内存有影响,你想做性能优化。等等等等,一句话,当你想深入了解java对象在内存中,如何存储,或者每个对象占用多大空间时,你会感谢这篇文章
本文主要分析jvm中的情况,实验环境为64位window10系统、JDK1.8,使用JProfiler进行结论验证
很多描述以及 概念是基于你懂基本java知识的,如果你看起来有点吃力,要加油咯
本片更偏重验证,更多理论,请参考:https://segmentfault.com/a/1190000006933272
内存公式:Java对象的内存布 = 对象头(Header) + 实例数据(Instance Data) + 补齐填充(Padding)。
补齐填充:Java对象占用空间是8字节对齐的,即所有Java对象占用bytes数必须是8的倍数
Shallow Size
对象自身占用的内存大小,不包括它引用的对象。
针对非数组类型的对象,它的大小就是对象与它所有的成员变量大小的总和。当然这里面还会包括一些java语言特性的数据存储单元。
针对数组类型的对象,它的大小是数组元素对象的大小总和。
Retained Size
Retained Size=当前对象大小+当前对象可直接或间接引用到的对象的大小总和。(间接引用的含义:A->B->C, C就是间接引用)
换句话说,Retained Size就是当前对象被GC后,从Heap上总共能释放掉的内存。
不过,释放的时候还要排除被GC Roots直接或间接引用的对象。他们暂时不会被被当做Garbage。
基本数据类型占用
类型 | 占用空间 |
---|---|
boolean、byte | 1byte |
short、char | 2byte |
int、float | 4byte |
long、double | 8byte |
接下来用JProfiler验证:
1.新建一个空对象,观察空对象内存占用
public class TestObject {}
对象占用内存 16b,如图
结论:一般自建空对象占用内存 16b,16 = 12(Header) + 4(Padding)
2.在TestObj中新增一个 int 属性,观察对象内存占用
public class TestObj { private int i;}
对象占用内存 16b,如图
结论:int 占用 4b, 16 = 12(Header) + 4(int)
3.在TestObj中新增一个 long 属性,观察对象内存占用
public class TestObj { private long i;}
对象占用内存 24b,如图
结论:long 占用 8b, 24 = 12(Header) + 8(long) + 4(Padding)
其余基本类型可以参照以上自行验证,原理一样
包装类型占用
包装类(Boolean/Byte/Short/Character/Integer/Long/Double/Float)占用内存的大小 = 对象头大小 + 底层基础数据类型的大小。
包装类和其他引用类一样,会产生一个引用(reference)
类型 | 占用空间 |
---|---|
Boolean、Byte | 16byte |
Short、Char | 16byte |
Integer、Float | 16byte |
Long、Double | 24byte |
1.在TestObj中新增一个 Integer 属性,观察对象内存占用
public class TestObj {
private Integer i =128;
}
对象占用内存 32b,如图
结论:Integer 占用 16b, 32 = 12 (Header) + 16(Integer) + 4(reference)
特别的:-128~127 在常量池,只占用 4b,且不产生引用(reference)
2.在TestObj中新增一个 Long 属性,观察对象内存占用
public class TestObj {
private Long l = new Long(1);
}
对象占用内存 40b,如图
结论:Long 占用 24b, 40 = 12 (Header) + 24(Long) + 4(reference)
其余包装类型可以参照以上自行验证,原理一样
基本类型数组占用
64位机器上,数组对象的对象头占用24 bytes,启用压缩后占用16字节。比普通对象占用内存多是因为需要额外的空间存储数组的长度(普通16b-12b)。
对象数组本身的大小=数组对象头 + length * 存放单个元素大小
在TestObj中新增一个 char[] 属性,观察对象内存占用
public class TestObj {
private char[] c = {'a','b','c'};
}
char[] c占用内存 40b,如图
结论:char[3] 占用 24b, 24 = 40 - 16,24 = 16(Header) + 3 * 2(char) + 2(Padding)
封装类型数组占用
封装类型数组比基本类型的数组,需要多管理元素的引用
对象数组本身的大小=数组对象头+length * 引用指针大小 + length * 存放单个元素大小
在TestObj中新增一个 Integer[] 属性,观察对象内存占用
public class TestObj {
private Integer[] i = {128,129,130};
}
Integer[] i占用内存 80b,如图
结论:Integer[3] 占用 80b, 80 = 96 - 16, 80 = 16(Header) + 3 * 4 (reference)+ 3 * 16(Integer) +4(padding)
String占用内存
在TestObj中新增一个空 String 属性,观察对象内存占用
public class TestObj {
private String s = new String("");
}
对象占用内存 40b,如图
结论:String 本身占用 24b, 24 = 40 -16,也就是说空""也需要16b
注意:这里为什么要写String s = new String("")?请自己思考,不写会怎么样?
答:如果写成String s = “”,是不会再堆中开辟内存的,也就看不到String占用的空间,你看到的将会是下面的,至于为什么,都是因为final
来源:https://blog.csdn.net/qq_30054961/article/details/88345030
猜你喜欢
- 最近学习Spring,一直不太明白Srping的切面编程中的的argNames的含义,经过学习研究后,终于明白,分享一下需要监控的类:pac
- Java中PriorityQueue通过二叉小顶堆实现,可以用一棵完全二叉树表示。本文从Queue接口函数出发,结合生动的图解,深入浅出地分
- 一、几句话使用Gradle及其推荐的项目框架把密码等敏感数据放入gradle.properties不要自己写Http客户端,使用Volley
- 一、简介当我们没有在子类构造函数中写上 base(),默认会先调用父类中无参的构造函数,再调用子类。当在有参构造函数后写上base时,只调用
- StringBuilder内部是由多段char[]组成的半自动链表,因此频繁从中间修改StringBuilder,会将原本连续的内存分隔为多
- 本文实例讲述了JAVA中AES加密方法。分享给大家供大家参考。具体如下:java代码:KeyGenerator kg = KeyGenera
- 简介Android给我们提供了一种轻量级的异步任务类AsyncTask。该类中实现异步操作,并提供接口反馈当前异步执行结果及进度,这些接口中
- 1.前言string是属于引用类型的,这个大家都知道吧?但是平常在使用的过程中,发现它还是拥有一些值类型的特征的,这到底是为什么呢?原因就是
- 1. 简单工厂介绍简单工厂有一个具体的工厂类,可以生产不同的产品,属于创建型设计模式。注意:简单工厂模式 不属于23种设计模式之列2. 简单
- 在任何Java面试当中多线程和并发方面的问题都是必不可少的一部分。如果你想获得任何股票投资银行的前台资讯职位,那么你应该准备很多关于多线程的
- 前言用户注册功能是每一个系统的入口门面功能,很多人可能会以为很简单,不就是一个简单的CRUD吗?其实不然,要把前后端功能都做出来,页面跳转也
- MyBatis提供了 * 接口,我们可以实现自己的 * ,将其作为一个plugin装入到SqlSessionFactory中。 首先要说的是
- 本文实例为大家分享了C#实现计算器窗体程序的具体代码,供大家参考,具体内容如下功能设计1、计算器中,添加 0-9 共十个数字键。2、计算器中
- 在Java中经常会涉及到对象数组的排序问题,那么就涉及到对象之间的比较问题。通常对象之间的比较可以从两个方面去看:第一个方面:对象的地址是否
- 本文实例讲述了Java实现二叉树的深度优先遍历和广度优先遍历算法。分享给大家供大家参考,具体如下:1. 分析二叉树的深度优先遍历的非递归的通
- 在使用AbstractRoutingDataSource配置多数据源时,发现使用@aspect配置的DataSourceSwitchAspe
- 在很多场景中我们需要验证时间日期的是否属于正确的格式,验证时间是否符合常规的。1、验证 yyyy-MM-dd HH:mm:dd 格式的日期S
- Map集合和Collection集合的区别Map集合是有Key和Value的,Collection集合是只有Value。Collection
- resultType和resultMap只能有一个成立,resultType是直接表示返回类型的,而resultMap则是对外部Result
- 本文实例为大家分享了java实现加减乘除计算器的具体代码,供大家参考,具体内容如下代码import java.awt.*;import ja