前一段时间在看关于如何测定JVM中对象占用的内存,涉及到JVM堆中对象结构相关知识。
在网上看到一篇比较不错的文章,分享出来,希望对看到的人有所帮助。
---------------------------------
原文地址:http://www.javamex.com/tutorials/memory/object_memory_usage.shtml
在JVM堆中,每个对象由四个域构成:
1、对象头,占用很少的字节,表述Object当前状态的信息:
“对象头”的作用是用来记录一个对象的实例名字、ID和实例状态(例如,当前实例是否“可到达”,或者当前锁的状态等等)。
注:java在64bit模式下开启指针压缩,比32bit模式下头部会大4byte(_mark区域变成8byte,_class区域被压缩),即12字节。
如果没有开启指针压缩,头部会大8byte(_mark和_class都会变成8byte),即16字节。
jdk1.6参数-XX:+UseCompressedOops(指针压缩),在32G内存下默认会自动开启这个参数。
2、基本类型域占用的空间(原生域指int、boolean、short等):
boolean、byte占用1 byte,char、short占用2 bytes,int、float占用4bytes,long、double占用 8 bytes。
3、引用类型域占用的空间(引用类型域指 其他对象的引用,每个引用占用4个字节):
每个引用类型占用4bytes。
注:对于组合对象(比如String对象)而言,它不会把“被组合”的对象(比如:String中的char[]数组对象)作为自身空间,而是开辟一个引用类型空间来引用组合对象。组合关系是一种对象之间的相互引用,引用和被引用的对象之间相互分离的。在物理空间上,避免了一个对象和其成员对象在空间上必须连续的问题,而是一种逻辑上的Link。同样,被引用对象同样以8字节的倍数进行对齐。
4、填充物占用的空间:
在Hotspot中,每个对象占用的总空间是以8的倍数计算的,对象占用总空间(对象头+声明变量)不足8的倍数时候,自动补齐。而这些被填充的空间,我们可以称它为“填充物”。
注:对象采用8字节对齐的方式是不论32bit还是64bit都是一样的。
另外,如果对象是数组:则至少占用12字节(多出4个字节保存数组长度),同时数组也会以8字节自动补齐。
意义:
现在了解了JVM堆中对象的结构,就能在日常编程中对创建的对象估算其占用内存大小,以便进行优化。
例如:
1、现在有500个字节存入数组,可以写作
byte[] bytes = new byte[500]; //每个元素存储为一个byte.
或者
Byte[] bytes = new Byte[500]; //每个元素存储为一个Byte.
(1)首先计算一下new byte[500];占用内存大小:
8+4+1*500 = 512 byte
前12个字节是数组中8byte的空Object和4byte的数组长度,数组中每个元素占用1byte。
(2)然后计算new Byte[500];占用内存大小:
8+4+4*500+16*500=10012 byte -> 对齐 = 10016 byte
前12个字节是数组中8byte的空Object和4byte的数组长度,数组中包含500个Byte的引用(4*500),同时由于
每个元素都是Byte对象,共500个(16*500,一个Byte对象要以8字节对齐,所以为16byte)。
以上结果都经过SizeOf工具验证,同时可以参考我的blog:
使用SizeOf测定JVM中对象占用内存 http://shensy.iteye.com/blog/1765760
结论:使用byte数组存储占用内存0.5k,使用Byte数组存储占用内存近10k。
2、编程时,能用Boolean.TRUE或FALSE就不要用new Boolean(),能用Integer.valueOf的时候就不要用new Integer(),其它各数据类型也都如此。原因是这些方法会存储一部分数值放在一块共享内存区域作为缓存使用。详解请见我的blog:
JVM对象占用内存计算工具--SizeOf源码分析 http://shensy.iteye.com/blog/1858987
结束语:
理解了JVM堆中对象结构,对于日常Java编程中内存优化会有很大帮助。
相关推荐
堆是所有对象实例的存储场所,是JVM中最大的一块内存;方法区则存储类的元数据,如类的版本、字段、方法信息等。 类加载机制是JVM的另一重要概念,它包括加载、验证、准备、解析和初始化五个阶段。类加载过程确保了...
1. **理解内存区域与内存区域异常**:学习JVM的内存结构,包括程序计数器、Java堆、虚拟机栈、本地方法栈、方法区和运行时常量池等,并了解它们发生异常的原因。 2. **掌握Java堆内存溢出异常的测试**:通过编写...
1. **堆(Heap)**:这是JVM中最大的一块内存区域,用于存储对象实例。所有通过`new`关键字创建的对象以及数组都会被分配到堆中。堆内存分为新生代(Young Generation)、老年代(Old Generation)和持久代...
- **堆**:所有对象实例以及数组都在这里分配内存,是JVM中最大的一块内存区域,支持垃圾回收。 - **栈**:每个线程都有一个独立的栈,用于存储方法调用的帧,包含局部变量表、操作数栈、动态链接和方法返回地址。...
Java虚拟机(JVM)是Java程序运行的基础,它将内存划分为多个区域,其中堆内存(Heap)和栈内存(Stack)是最基础且重要的两个部分。了解它们的区别对于优化程序性能至关重要。 栈内存主要用于存储程序运行过程中的...
本文将详细探讨Java对象在JVM中的创建过程以及其内存布局,帮助读者更深入地理解Java对象是如何在内存中产生的。 #### 二、对象的创建 Java对象是由类实例化的结果,当我们使用`new`关键字创建一个对象时,实际上...
Java优化编程是提升Java应用程序性能的关键技术,涵盖了多个方面的知识,包括代码优化、内存管理、垃圾回收、并发处理以及JVM参数调优等。以下是对这些主题的详细讲解: 1. **代码优化**:编写高效的Java代码是优化...
- **堆**:所有对象实例都在堆中分配内存,是JVM内存中最大的一块区域,进行垃圾收集的主要区域。 - **虚拟机栈**:每个线程都有一个独立的虚拟机栈,用于存储局部变量、操作数栈、动态链接等。 - **本地方法栈**...
了解JVM内存结构是理解内存溢出的关键。 #### 二、JVM运行时数据区域 - **程序计数器(Program Counter Register)**:当前线程所执行的字节码的行号指示器。每条线程拥有独立的程序计数器,因此它属于线程私有区域...
在Java中,堆内存和栈内存的比较可以从功能和性能两方面来看。栈内存执行效率高,但存储空间有限且分配/释放速度快;堆内存提供更大的灵活性,可以动态分配和释放,适合存储大型对象和数组,但分配和回收的效率较低...
- 堆内存:所有对象实例都在堆中分配内存,是垃圾收集的主要区域。 - 栈内存:每个线程都有自己的程序计数器、虚拟机栈、本地方法栈,用于执行方法调用。 - JIT编译器:JVM中的Just-In-Time编译器,将热点代码...
- **对象引用**:为了访问堆内存中的对象,可以在栈内存中创建一个引用变量,该变量的值为对象在堆内存中的地址。 - **垃圾回收**:Java虚拟机(JVM)会自动管理堆内存,通过垃圾回收机制(Garbage Collection, GC)...
Java虚拟机(JVM)内存管理是Java编程中不可或缺的一部分,尤其在面试中常常成为考察的重点。以下是对JVM内存结构、垃圾回收机制及其相关面试问题的详细解答: 1. JVM内存区域: - **堆内存**:存放所有的Java对象...
堆内存用于存储对象实例,而栈内存则用于存储方法调用时的局部变量。此外,还有方法区(或称为永久代)存放类的元数据,包括类信息、常量池等。Java的垃圾收集机制自动管理堆内存,通过不同类型的垃圾收集器来实现...
10. **JVM内存分析**:通过MAT(Memory Analyzer Tool)等工具进行堆dump分析,找出内存中的对象引用关系,定位内存泄漏的根源。 这份思维笔记会深入探讨以上各个知识点,通过实例和案例分析,帮助开发者从理论到...
在Java编程语言中,了解对象内存大小是优化内存使用、提高程序性能的关键步骤。当我们谈论“Java对象内存大小”时,我们通常指的是一个Java对象在内存中占据的空间,包括对象头、实例字段以及可能的对齐填充。这个...
2. **内存管理**:JVM内存主要分为堆内存和栈内存,其中堆内存用于对象实例的存储,栈内存则处理方法调用。了解内存分配、GC(Garbage Collection)策略以及内存泄漏的检测和预防对于避免程序运行缓慢至关重要。 3....
这个压缩包文件"JVM优化3(Tomcat参数调优,JVM参数调优,jvm字节码,代码优化).zip"显然包含了关于如何优化Java应用程序运行效率的四个主要方面:Tomcat服务器的参数调整、JVM参数调优、JVM字节码理解和优化以及代码...