`
longdick
  • 浏览: 584356 次
  • 性别: Icon_minigender_1
  • 来自: 0
社区版块
存档分类
最新评论

图解JVM内存模型

    博客分类:
  • JVM
阅读更多

/**

*  转载请注明作者longdick    http://longdick.iteye.com

*

*/

 

Java 的内存模型由3个代组成,各个代的默认排列有如下图(适用JDK1.4.*  到 JDK6):


Java 的内存模型分为

Young(年轻代)

Tenured(终身代)

Perm(永久代)

 

有些旧版本也叫作

New

Old

Perm

 

叫法不同,表达的意思却是基本相同。

 

注意Young(年轻代)还可以分为Eden区和两个Survivor区(from和to,这两个Survivor区大小严格一至),新的对象实例总是首先放在Eden区,Survivor区作为Eden区和 Tenure(终生代)的缓冲,可以向 Tenure(终生代)转移活动的对象实例。

Tenure(终生代)中存放生命周期长久的实例对象,但并不是如它的名字那样是终生的,里面的对象照样会被回收掉。

Young和Tenure共同组成了堆内存。

 

Perm(永久代)则是非堆内存的组成部分。主要存放加载的Class类级对象如class本身,method,field等等。

 

有同学可能已经注意到了,每个代都有的Virtual区又是什么?

 

我们知道有一些参数可以影响以上各代的大小。

在JVM启动时,就已经保留了固定的内存空间给Heap内存,这部分内存并不一定都会被JVM使用,但是可以确定的是这部分保留的内存不会被其他进程使用。这部分内存大小由 -Xmx 参数指定。

 

而另一部分内存在JVM启动时就分配给JVM,作为JVM的初始Heap内存使用。影响这个的参数是 -Xms ,如果 -Xms 指定的值比-Xmx 的小,那么两者的差值就是Virtual内存值。随着程序的运行,Eden区、 Tenured区和Perm区会逐渐使用保留的Virtual空间。

 

如果没有具体指定,初始和最大堆内存将根据机器的内存计算得出。参数DefaultInitialRAMFraction DefaultMaxRAMFraction 会影响最终的结果,如下表所示:

 


Formula Default
initial heap size memory / DefaultInitialRAMFraction memory / 64
maximum heap size MIN(memory / DefaultMaxRAMFraction, 1GB) MIN(memory / 4, 1GB)

 

可以看到堆内存默认值最大不会超过1G。

 

JVM会根据堆内存的使用情况自动决定何时扩张和缩减实际堆内存的大小,可以用VM参数 -XX:MinHeapFreeRatio=<minimum> -XX:MaxHeapFreeRatio=<maximum> 使用堆内存空闲百分比来定义,一般在32位机器上的默认值如下:

 

Parameter Default Value
MinHeapFreeRatio 40
MaxHeapFreeRatio 70
-Xms 3670k
-Xmx 64m

 

 

当空闲堆内存所占堆内存百分比低于40%,JVM就会试图扩张堆内存空间;当空闲堆内存所占堆内存百分比高于70%,JVM就会试图压缩堆内存空间。

ps:以上默认值在不同平台会有不同的值,如果是64位系统,这些值一般需要扩张30%,来容纳在64位系统下变大的对象。

 

加上-XX:NewRatio=3 意味着 young(年轻代) 和 tenured(终生代)的比率是1:3,也就是说,eden区和survivor区容量之和将占总堆内存的1/4。

 

加上-XX:SurvivorRatio=6 设置eden区和 其中一个survivor space的比率是1:6,也就是说,其中一个survivor space占年轻代1/8的容量 (可以想想为什么不是1/7)。

 

另外还有 -XX:NewSize -XX:MaxNewSize   指定年轻代的初始值和最大值。

32位系统下默认值如下:

 


Default Value Parameter Client JVM Server JVM
NewRatio 8 2
NewSize 2228K 2228K
MaxNewSize not limited not limited
SurvivorRatio 32 32

 

 

延伸阅读:

图解JVM在内存中申请对象及垃圾回收流程

三种GC大揭秘

参考资料:

http://java.sun.com/javase/technologies/hotspot/gc/gc_tuning_6.html

21
4
分享到:
评论
12 楼 zhoujianboy 2016-04-19  
推荐一篇文章 JVM内存模型和JVM参数的关系:http://www.haonanji.cn/408.html
11 楼 hordaway 2014-02-21  
错误百出
10 楼 hordaway 2014-02-21  
加上-XX:SurvivorRatio=6
设置eden区和 其中一个survivor space的比率是1:6,也就是说,其中一个survivor space占年轻代1/8的容量 (可以想想为什么不是1/7)。

这段话是错误的

正确的应该是
加上-XX:SurvivorRatio=6
设置eden区和 其中一个survivor space的比率是6:1,也就是说,其中一个survivor space占年轻代1/8的容量 (可以想想为什么不是1/7)。
8 楼 henry0609 2013-08-07  
hacker_zxf 写道
“Perm(永久代)则是非堆内存的组成部分”这句话是不是不对?
perm是-Xmx中的一部分,-Xmx是指最大的heap数值


我也有相同的疑惑。既然 Young 和 Tenured 共同组成了 Heap, -Xmx 参数是配置最大堆内存的, 而 Perm 是非堆内存,那 -Xmx 应该不影响 Perm 才对啊。

另外, 楼主说: Virtual = -Xmx - -Xms, 说的都是堆内存,

后面又说:“随着程序的运行,Eden区、 Tenured区和Perm区会逐渐使用保留的Virtual空间。”

既然 Perm 是非堆内存,它怎么会用到 Virtual 空间呢?

还望楼主指教。
7 楼 337240552 2012-08-29  
写的很好, ,适合初学者
6 楼 diyunpeng 2012-04-11  
在JVM启动时,就已经保留了固定的内存空间给Heap内存,这部分内存并不一定都会被JVM使用,但是可以确定的是这部分保留的内存不会被其他进程使用。这部分内存大小由 -Xmx
参数指定。
junefsh 写道
在JVM启动时,就已经保留了固定的内存空间给Heap内存,这部分内存并不一定都会被JVM使用,但是可以确定的是这部分保留的内存不会被其他进程使用。这部分内存大小由 -Xmx
参数指定。


兄弟,你自己测试了没,别误导人


我可以接受你的观点,但是我建议还有最好拿出官方的文档来证明确实是Virtual是你说的那样,我印象中默认JDK在32位机器上ms是16M,mx是64M,但是是不是这个48M就是Virtual,起码看了很多Oracle和Sun的文档,我没有注意过。
5 楼 junefsh 2011-10-29  
在JVM启动时,就已经保留了固定的内存空间给Heap内存,这部分内存并不一定都会被JVM使用,但是可以确定的是这部分保留的内存不会被其他进程使用。这部分内存大小由 -Xmx
参数指定。


兄弟,你自己测试了没,别误导人
4 楼 hacker_zxf 2011-08-05  
我错了 perm 应该是-XX:MaxPermSize的一部分,-XX:MaxPermSize 减去 -XX:PermSize 为perm的virtual 区域,这边说的有点奇异
3 楼 hacker_zxf 2011-08-05  
“Perm(永久代)则是非堆内存的组成部分”这句话是不是不对?
perm是-Xmx中的一部分,-Xmx是指最大的heap数值
2 楼 lee_ty 2009-10-21  
感谢楼主,形象具体,清晰易懂
1 楼 whaosoft 2009-09-23  
受教了哦 感谢lz

相关推荐

    JVM图解-JVM指令-JVM原型图.rar

    以上只是JVM众多知识中的一部分,实际上,JVM涉及的领域还包括内存模型、线程管理、异常处理、类加载策略等。理解JVM的工作原理对于编写高效、稳定的Java程序至关重要。通过研究这个压缩包中的资源,你可以更深入地...

    JVM技术,反射与动态代理

    3. 内存模型:JVM内存包括堆、栈、方法区、本地方法栈和程序计数器等区域,其中堆是所有线程共享的,用于存储对象实例;栈则对应每个线程,存储局部变量和方法调用信息。 4. 垃圾回收:JVM自动管理内存,通过垃圾...

    java中多态的内存分析

    首先,了解Java内存模型至关重要。Java程序运行时主要涉及四种内存区域:程序计数器、虚拟机栈、本地方法栈、堆和方法区(在Java 8及以后版本中,方法区被元空间取代)。 1. **程序计数器**:每个线程都有一个独立...

    6.1.1.JVM前奏篇笔记1

    它们在内存模型、垃圾收集、性能优化等方面有所不同。图解可以帮助我们更直观地理解这些概念,例如类加载机制的图解可以展示类从加载到初始化的完整流程。 总的来说,了解JVM的工作原理、类文件的结构以及编译过程...

    06-Java基础(数组-内存图解)

    在内存模型中,数组被存储在堆内存(Heap Memory)中,每个数组元素都有一个连续的地址。数组的首地址(即数组名所指向的地址)可以看作是数组的引用。当我们通过索引访问数组元素时,例如`numbers[0]`,Java虚拟机...

    【Java面试+Java学习指南】一部分大部分Java招聘所需要掌握的核心知识

    JVM内存模型 性能调优、线上问题排查 类加载机制详解 垃圾回收机制 垃圾回收器、垃圾回收算法 ARM与多线程 多线程基础知识 常见关键字 多线程锁机制 线程池知识点 常见的JUC工具类 多线程经典面试题 常用工具集 ...

    图解Spark核心技术与案例实战&&elasticsearch;-the-definitive-guide&&JVM;高级特性与最佳实践(最新第二版)

    5. **JVM最佳实践**:书中可能提供了一系列在实际开发中应该遵循的JVM使用建议,如选择合适的垃圾收集器、避免内存泄漏、监控JVM状态、诊断性能问题等。通过遵循这些最佳实践,可以确保应用程序在大规模数据处理时...

    Java学习指南,涵盖大部分Java程序员所需要掌握的核心知识

    JVM内存模型 性能调优、线上问题排查 类加载机制详解 垃圾回收机制 垃圾回收器、垃圾回收算法 并发与多线程 多线程基础知识 常见关键字 多线程锁机制 线程池知识点 常见的JUC工具类 多线程经典面试题 常用工具集 ...

    Java内存结构.pdf

    Java内存模型主要分为以下几个部分:堆(Heap)、栈(Stack)、方法区(Method Area)、程序计数器(Program Counter Register)以及本地方法栈(Native Method Stack)。本文将重点介绍其中的堆、栈、方法区以及...

    面试快餐几个图一目了然-临阵补仓.zip

    面试中,Java相关的知识点可能包括基础语法、集合框架、多线程、异常处理、JVM内存模型、IO流、网络编程以及设计模式等。 【压缩包子文件的文件名称列表】虽然未提供具体的文件名,但根据上述信息,我们可以推测...

    java学习图解

    12. **Java虚拟机(JVM)**: JVM是Java程序的执行环境,理解其内存模型(如堆、栈、方法区等)和垃圾回收机制对于性能优化至关重要。 13. **Java集合框架的高级特性**: 如并发容器(如ConcurrentHashMap)和Stream ...

    8张图解java

    **运行时内存模型**: - **程序计数器**: 指向当前线程正在执行的字节码指令。 - **虚拟机栈**: 保存每个线程的局部变量和操作数栈。 - **本地方法栈**: 与虚拟机栈类似,但用于本地(Native)方法。 - **Java堆**:...

    《图解JSP环境安装配置(CHM) 》

    2. **设置JVM内存**:在`bin/catalina.sh`(或`catalina.bat`)脚本中,添加或修改`JAVA_OPTS`以调整JVM的内存分配,例如`JAVA_OPTS="-Xms512m -Xmx1024m"`。 3. **部署目录配置**:默认情况下,Tomcat会监视`...

    Java并发编程全景图.pdf

    Java并发编程是Java语言中最为复杂且重要的部分之一,它涉及了多线程编程、内存模型、同步机制等多个领域。为了深入理解Java并发编程,有必要了解其核心技术点和相关实现原理,以下将详细介绍文件中提及的关键知识点...

    Java 面试资料

    至于"java面试相关资料"这个文件,可能包含各种面试题目和答案,包括但不限于基础语法、集合框架、多线程、异常处理、IO流、设计模式、JVM内存模型、Spring框架、微服务等相关知识。这些都是Java开发者必须掌握的...

    java知识点.zip

    "java基础图解"则可能包含了一系列关于Java基础知识的图表,如内存模型、垃圾回收机制、JVM(Java虚拟机)的工作原理,以及如何优化代码性能。理解这些可以帮助开发者写出更高效、更稳定的程序。 进一步深入,我们...

    Java程序员经典书籍推荐

    4. **JVM内部结构剖析**:深入探讨JVM的各个组件如类加载机制、内存模型等的工作原理。 5. **高级特性详解**:比如垃圾回收算法、JIT编译等,这些都是提升Java应用性能的关键因素。 此书不仅内容全面、深入浅出,...

    spribgoot-tomcat代码.zip

    除了上述策略,还可以通过调整JVM参数,如增大堆内存、开启G1垃圾收集器等方式优化性能。同时,使用缓存技术如Redis,可以减少数据库访问,提升响应速度。对于数据库访问,使用连接池如HikariCP,可以更高效地管理...

Global site tag (gtag.js) - Google Analytics