`
benson76
  • 浏览: 34563 次
  • 性别: Icon_minigender_1
  • 来自: 广州
社区版块
存档分类
最新评论

Java对象大小探究

    博客分类:
  • JAVA
 
阅读更多

参考:

http://langyu.iteye.com/blog/1167581

http://www.javamex.com/tutorials/memory/object_memory_usage.shtml

http://www.codeinstructions.com/2008/12/java-objects-memory-structure.html

http://www.javamex.com/tutorials/memory/array_memory_usage.shtml


下载classmexer.jar

$ javac -cp ./classmexer.jar BytesDemo.java
$ java -javaagent:classmexer.jar BytesDemo


MemoryUtil.memoryUsageOf(demo)  计算demo对象的堆大小

MemoryUtil.deepMemoryUsageOf(demo)  计算demo对象的堆大小,以及他所引用的对象的大小

class Base {
  long a;
}
public class BytesDemo {
        Base b = new Base();
        public static void main(String[] args) {
            BytesDemo demo = new BytesDemo();
            System.out.println("Object size : " + MemoryUtil.memoryUsageOf(demo));  
            System.out.println("Object size : " + MemoryUtil.deepDemoryUsageOf(demo));    
        }
    }






 输出为:16,32




      从上面的一张图上看到了两个名词: shallow size 和 retained size 。Shallow表示本对象自身的大小是多少。本对象可能会直接或间接引用其它很多对象,如果被引用的对象仅仅被本对象所引用,那么当本对象无用被GC时,那么本对象所引用的对象也会被GC。Retained就表示如果当本对象被GC时能相关地减少的内存量。


shallow size就是MemoryUtil.memoryUsageOf()的值,

我认为大小关系:shallow size<=retained size<=MemoryUtil.deepDemoryUsageOf()


     classmexer对于2种情况计算并不准确,例如Threalocal等不再堆中存放的对象,对象是否包含synchronized锁,是否在gc。

We'll generally be talking about the memory taken up on the heap by a given object under "normal circumstances". A couple of minor complications that we'll gloss over are that:

  • in some cases, a JVM may not put an object on the heap at all: for example, a small thread-local object could in principle be held entirely on the stack or in registers and not "exist" strictly speaking as a Java heap object;
  • the overall memory impact of an object can depend on its current state : for example, whether its synchronization lock is contended , or whether it is being garbage collected (such extra "system" data is not necessarily allocated on the Java heap, however).

对象占的内存包含一下:

In general, the heap memory used by a Java object in Hotspot consists of:

  • an object header , consisting of a few bytes of "housekeeping" information;
  • memory for primitive fields , according to their size (see below);
  • memory for reference fields (4 bytes each);
  • padding : potentially a few "wasted" unused bytes after the object data, to make every object start at an address that is a convenient multiple of bytes and reduce the number of bits required to represent a pointer to an object.


      每个对象的物理存储可以分为两部分:Header及该对象的所有对象属性值(不包括 static属性)

      (1) Hotspot VM限定每个对象header是2个word,word是JVM内部的存储最小单位,当前Hotspot定义的word大小是4字节,所以header共是8个字节。Header中应当包含着本对象的hashCode,对象锁及与GC相关的生存周期信息等。

      Instances of an object on the Java heap don't just take up memory for their actual fields. Inevitably, they also require some "housekeeping" information, such as recording an object's class, ID and status flags such as whether the object is currently reachable, currently synchronization-locked etc.

In Hotspot:
  • a normal object requires 8 bytes of "housekeeping" space;
  • arrays require 12 bytes (the same as a normal object, plus 4 bytes for the array length).

Other JVMs probably have a similar object overhead.

      (2) 对象属性分为两部分:基本类型属性与对象引用。 Java事先定义了所有基本类型的占用位数 ,如下表:


     基本类型对象属性依上表占用着堆内存,而每个对其它对象的引用是规定占用一个word,也就是4个字节。 像上面第二个目标类中新增加一个对象引用cache,那么这个引用属性就只占用4个字节。

    正常的对象引用也有两种:普通对象与数组。数组也是正常对象,只不过,它除了header外还有4个字节表示当前数组的长度是多少,那么我们也可以认为数组对象的header长度就是12个字节了。

  • in Java, an array is a special type of object ;
  • a multi-dimensional array is simple an array of arrays ;
  • for example, in a two-dimensional array , every row is a separate array object .

      A single-dimension array is a single object. As expected, the array has the usual object header . However, this object head is 12 bytes to accommodate a four-byte array length . Then comes the actual array data which, as you might expect, consists of the number of elements multiplied by the number of bytes required for one element, depending on its type.


      在这里要特别强调的是如果某个普通对象就包含一个byte属性,那么它的对象大小应该是9个字节。而JVM为了malloc与gc方便,指定分配的每个对象都需要是8字节的整数倍 ,那么对象大小就不再是9个字节,而应该是16个字节。

In Hotspot, every object occupies a number of bytes that is a multiple of 8 . If the number of bytes required by an object for its header and fields is not a multiple 8, then you round up to the next multiple of 8 .

This means, for example, that:

  • a bare Object takes up 8 bytes;
  • an instance of a class with a single boolean field takes up 16 bytes: 8 bytes of header, 1 byte for the boolean and 7 bytes of "padding" to make the size up to a multiple of 8;
  • an instance with eight boolean fields will also take up 16 bytes: 8 for the header, 8 for the booleans; since this is already a multiple of 8, no padding is needed;
  • an object with a two long fields, three int fields and a boolean will take up:
    • 8 bytes for the header;
    • 16 bytes for the 2 longs (8 each);
    • 12 bytes for the 3 ints (4 each);
    • 1 byte for the boolean;
    • a further 3 bytes of padding, to round the total up from 37 to 40, a multiple of 8.


多维数组的内存占用情况:

For example, let's consider a 10x10 int array. Firstly, the "outer" array has its 12-byte object header followed by space for the 10 elements. Those elements are object references to the 10 arrays making up the rows. That comes to 12+4*10=52 bytes, which must then be rounded up to the next multiple of 8, giving 56. Then, each of the 10 rows has its own 12-byte object header, 4*10=40 bytes for the actual row of int s, and again, 4 bytes of padding to bring the total for that row to a multiple of 8. So in total, that gives 11*56=616 bytes. That's a bit bigger than if you'd just counted on 10*10*4=400 bytes for the hundred "raw" int s themselves.


分享到:
评论

相关推荐

    [Java拾遗]Java对象大小探究

    NULL 博文链接:https://langyu.iteye.com/blog/1167581

    狂神说JVM探究.rar

    【狂神说JVM探究】是一份集合了多种格式的学习资料,主要涵盖了Java虚拟机(JVM)的基础知识。这份资料出自B站上的【狂神说Java】系列教程,为快速入门JVM提供了详实的笔记。以下是根据这些资源可能包含的一些关键...

    Java内存不足PermGen space错误探究.pdf

    1. 堆(Heap):这是Java对象的主要存储区域,根据对象的生命周期,堆又分为新生代(Young Generation)、老年代(Old Generation)和持久代(Permanent Generation)。 2. 栈(Stack):每个线程都有自己的程序...

    33、JVM探究.pdf

    【JVM探究】 Java虚拟机(JVM)是Java编程语言的核心组成部分,它负责执行Java程序,提供了一个跨平台的运行环境。理解JVM的工作原理对于优化Java应用程序性能至关重要。以下是一些关于JVM的关键知识点: 1. **JVM...

    Java垃圾收集必备手册

    在这篇手册中,我们将深入探究 Java 垃圾收集的基础知识,包括垃圾收集的定义、垃圾收集算法、垃圾收集器的实现、垃圾收集优化等内容。 垃圾收集的定义 垃圾收集(Garbage Collection,简称 GC)是指在 Java 语言中...

    Java工作实用篇.pdf

    Java虚拟机深入探究 - **内容覆盖**: - Java虚拟机的基本架构。 - 平台无关性的实现原理。 - 安全机制的设计。 - 网络移动性的支持。 - **学习价值**: - 深入理解Java程序的执行环境。 - 优化程序性能和资源...

    mat java 分析 文件 dump

    - **Shallow Size**:表示该类所有对象所占用的实际内存大小(不包括其引用的对象所占用的空间)。 - **Retained Size**:表示该类所有对象及其引用的对象所占用的总内存大小。 **示例分析:** 根据提供的部分内容...

    JAVA源码Java中的SSL及HTTPS协议实例源码

    在深入探究Java语言实现SSL(Secure Sockets Layer,安全套接层)及HTTPS(Hyper Text Transfer Protocol Secure,超文本传输协议安全版)协议的过程中,我们首先需要了解SSL和HTTPS的基本概念及其重要性。...

    Java JVM Instruction Set

    在设计JVM时,考虑到Java语言的特点,如面向对象、自动垃圾回收等特性,JVM的设计需要能够支持这些特性。 ##### 如何检查JVM代码 可以使用`javap`工具来反汇编已编译的Java类文件,查看字节码指令。例如,如果想要...

    java-jdk源码学习

    源码学习则是对这些组件内部实现的探究。 1. **Java编译器(javac)** - Java源代码是用`.java`后缀的文件编写的,`javac`负责将这些源代码编译成字节码(`.class`文件),这是跨平台运行的基础。 - `javac`的...

    怎样用Jvm处理Java数组.doc

    Java虚拟机(JVM)是Java程序的核心执行环境,它负责解释和执行字节码,同时也管理内存和类加载。在处理Java数组时,有一些关键的特性...理解这一点有助于我们更好地理解和使用Java数组,以及深入探究JVM的工作原理。

    JAVA集合介绍

    - `transient` 用于标记对象中的某个属性,使其不在序列化的过程中被保存。 - 在序列化过程中,`transient` 声明的属性会被忽略,只保留非 `transient` 属性。 ##### 2.2 构造函数 `ArrayList` 提供了多种构造...

    Java面试宝典PDF

    2. 虚拟机参数调整:学习如何设置堆大小、新生代与老年代的比例、GC策略等参数。 3. 内存泄漏检测:识别内存泄漏的迹象,使用工具进行监控和诊断。 八、Spring框架 1. Spring核心:理解依赖注入(DI)和控制反转...

    java经典小程序.doc

    Java作为一种面向对象的编程语言,自问世以来就以其平台独立性、丰富的类库及简洁的语法特性,在程序开发领域中占据了重要地位。尤其对于初学者而言,通过编写一些经典的小程序,不仅可以迅速掌握Java编程的基本概念...

    JVM实战-对象访问与内存溢出异常解析

    2. **掌握Java堆内存溢出异常的测试**:通过编写程序不断地创建对象,直至消耗完所有可用的堆内存,从而触发`OutOfMemoryError`异常。 3. **掌握虚拟机栈和本地方法栈内存溢出异常的测试**:探究如何通过递归调用或...

    CardGames:Java 中的纸牌和瓷砖游戏集合

    《Java中的纸牌和瓷砖游戏集合》 在Java编程领域,开发游戏是一种常见的实践,它不仅可以帮助开发者提高编程技能,还能让学习过程充满...无论是对Java编程感兴趣的学生还是经验丰富的开发者,这个项目都值得深入探究。

    MAT工具(基于Eclipse的内存分析工具)+MAT工具使用说明.docx

    使用内存分析工具从众多的对象中进行分析,快速的计算出在内存中对象的占用大小,看看是谁阻止了垃圾收集器的回收工作,并可以通过报表直观的查看到可能造成这种结果的对象。 二 使用MAT意义 当服务器应用占用了...

    借HSDB来探索HotSpot VM的运行时数据1

    在启动jdb时,通过`-XX:+UseSerialGC`选项启用串行垃圾收集器,并使用`-Xmx10m`限制Java堆大小为10MB,这样可以更好地控制和理解内存行为。通过`stop in`命令在`Test.fn`方法入口处设置断点,然后使用`run`命令启动...

    深入浅出jvm虚拟机视频大全(jvm性能调优+内存模型+虚拟机原理)

    ### JVM虚拟机原理探究 #### 1. 类加载机制 类加载过程包括加载、验证、准备、解析和初始化五个阶段。其中,加载阶段负责读取类文件,并将其转换成二进制流;验证阶段确保类文件符合JVM规范;准备阶段为类变量分配...

Global site tag (gtag.js) - Google Analytics