参考:
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.
分享到:
相关推荐
NULL 博文链接:https://langyu.iteye.com/blog/1167581
【狂神说JVM探究】是一份集合了多种格式的学习资料,主要涵盖了Java虚拟机(JVM)的基础知识。这份资料出自B站上的【狂神说Java】系列教程,为快速入门JVM提供了详实的笔记。以下是根据这些资源可能包含的一些关键...
1. 堆(Heap):这是Java对象的主要存储区域,根据对象的生命周期,堆又分为新生代(Young Generation)、老年代(Old Generation)和持久代(Permanent Generation)。 2. 栈(Stack):每个线程都有自己的程序...
【JVM探究】 Java虚拟机(JVM)是Java编程语言的核心组成部分,它负责执行Java程序,提供了一个跨平台的运行环境。理解JVM的工作原理对于优化Java应用程序性能至关重要。以下是一些关于JVM的关键知识点: 1. **JVM...
在这篇手册中,我们将深入探究 Java 垃圾收集的基础知识,包括垃圾收集的定义、垃圾收集算法、垃圾收集器的实现、垃圾收集优化等内容。 垃圾收集的定义 垃圾收集(Garbage Collection,简称 GC)是指在 Java 语言中...
Java虚拟机深入探究 - **内容覆盖**: - Java虚拟机的基本架构。 - 平台无关性的实现原理。 - 安全机制的设计。 - 网络移动性的支持。 - **学习价值**: - 深入理解Java程序的执行环境。 - 优化程序性能和资源...
- **Shallow Size**:表示该类所有对象所占用的实际内存大小(不包括其引用的对象所占用的空间)。 - **Retained Size**:表示该类所有对象及其引用的对象所占用的总内存大小。 **示例分析:** 根据提供的部分内容...
在深入探究Java语言实现SSL(Secure Sockets Layer,安全套接层)及HTTPS(Hyper Text Transfer Protocol Secure,超文本传输协议安全版)协议的过程中,我们首先需要了解SSL和HTTPS的基本概念及其重要性。...
在设计JVM时,考虑到Java语言的特点,如面向对象、自动垃圾回收等特性,JVM的设计需要能够支持这些特性。 ##### 如何检查JVM代码 可以使用`javap`工具来反汇编已编译的Java类文件,查看字节码指令。例如,如果想要...
源码学习则是对这些组件内部实现的探究。 1. **Java编译器(javac)** - Java源代码是用`.java`后缀的文件编写的,`javac`负责将这些源代码编译成字节码(`.class`文件),这是跨平台运行的基础。 - `javac`的...
Java虚拟机(JVM)是Java程序的核心执行环境,它负责解释和执行字节码,同时也管理内存和类加载。在处理Java数组时,有一些关键的特性...理解这一点有助于我们更好地理解和使用Java数组,以及深入探究JVM的工作原理。
- `transient` 用于标记对象中的某个属性,使其不在序列化的过程中被保存。 - 在序列化过程中,`transient` 声明的属性会被忽略,只保留非 `transient` 属性。 ##### 2.2 构造函数 `ArrayList` 提供了多种构造...
2. 虚拟机参数调整:学习如何设置堆大小、新生代与老年代的比例、GC策略等参数。 3. 内存泄漏检测:识别内存泄漏的迹象,使用工具进行监控和诊断。 八、Spring框架 1. Spring核心:理解依赖注入(DI)和控制反转...
Java作为一种面向对象的编程语言,自问世以来就以其平台独立性、丰富的类库及简洁的语法特性,在程序开发领域中占据了重要地位。尤其对于初学者而言,通过编写一些经典的小程序,不仅可以迅速掌握Java编程的基本概念...
2. **掌握Java堆内存溢出异常的测试**:通过编写程序不断地创建对象,直至消耗完所有可用的堆内存,从而触发`OutOfMemoryError`异常。 3. **掌握虚拟机栈和本地方法栈内存溢出异常的测试**:探究如何通过递归调用或...
《Java中的纸牌和瓷砖游戏集合》 在Java编程领域,开发游戏是一种常见的实践,它不仅可以帮助开发者提高编程技能,还能让学习过程充满...无论是对Java编程感兴趣的学生还是经验丰富的开发者,这个项目都值得深入探究。
使用内存分析工具从众多的对象中进行分析,快速的计算出在内存中对象的占用大小,看看是谁阻止了垃圾收集器的回收工作,并可以通过报表直观的查看到可能造成这种结果的对象。 二 使用MAT意义 当服务器应用占用了...
在启动jdb时,通过`-XX:+UseSerialGC`选项启用串行垃圾收集器,并使用`-Xmx10m`限制Java堆大小为10MB,这样可以更好地控制和理解内存行为。通过`stop in`命令在`Test.fn`方法入口处设置断点,然后使用`run`命令启动...
### JVM虚拟机原理探究 #### 1. 类加载机制 类加载过程包括加载、验证、准备、解析和初始化五个阶段。其中,加载阶段负责读取类文件,并将其转换成二进制流;验证阶段确保类文件符合JVM规范;准备阶段为类变量分配...