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

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

    博客分类:
  • JVM
阅读更多

/**

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

*

*/

 

先看一下JVM的内存模型:

 



 

从大的方面来讲,JVM的内存模型分为两大块:

 

永久区内存( Permanent space )和堆内存(heap space)。

 

栈内存(stack space)一般都不归在JVM内存模型中,因为栈内存属于线程级别。

每个线程都有个独立的栈内存空间。

 

Permanent space里存放加载的Class类级对象如class本身,method,field等等。

heap space主要存放对象实例和数组。

heap space由Old Generation和New Generation组成,Old Generation存放生命周期长久的实例对象,而新的对象实例一般放在New Generation。

New Generation还可以再分为Eden区(圣经中的伊甸园)、和Survivor区,新的对象实例总是首先放在Eden区,Survivor区作为Eden区和Old区的缓冲,可以向Old区转移活动的对象实例。

 

下图是JVM在内存空间(堆空间)中申请新对象过程的活动图(点击看大图):

没错,我们常见的OOM(out of memory)内存溢出异常,就是堆内存空间不足以存放新对象实例时导致。

 

永久区内存溢出相对少见,一般是由于需要加载海量的Class数据,超过了非堆内存的容量导致。通常出现在Web应用刚刚启动时,因此Web应用推荐使用预加载机制,方便在部署时就发现并解决该问题。

 

栈内存也会溢出,但是更加少见。

 

堆内存优化:

调整JVM启动参数-Xms  -Xmx   -XX:newSize -XX:MaxNewSize,如调整初始堆内存和最大对内存 -Xms256M -Xmx512M。 或者调整初始New Generation的初始内存和最大内存 -XX:newSize=128M -XX:MaxNewSize=128M。

 

永久区内存优化:

调整PermSize参数   如  -XX:PermSize=256M -XX:MaxPermSize=512M

 

栈内存优化:

调整每个线程的栈内存容量  如  -Xss2048K

 

 

最终,一个运行中的JVM所占的内存= 堆内存  +  永久区内存  +  所有线程所占的栈内存总和

 

 

  • 大小: 101.6 KB
  • 大小: 11.6 KB
39
0
分享到:
评论
15 楼 myNameIs-Sls 2013-12-27  
14 楼 ligf06 2013-09-05  
非常好,很需要看这个模型!
13 楼 diyunpeng 2012-04-11  
mxswl 写道
mxswl 写道
我只知道最经典的运行时环境对内存的分块如下: 代码区 + 栈区 + 堆区.
JVM在这最基本的模型上搞出了不少花样啊.

另外我觉得字符串的引用变量(其实也就是一普通引用)的确是在栈上的,但是java里的字符串也是一个数据结构(String类),其里面真正的数据  char[] 就是放在堆区上了.

笔误,char[] 是在常量区. 不在堆上.


这里我觉得看了N多文章,都是理论说明,谁真正能够证明确实存在于Perm上,除非查看JDK代码,在理论与实践上,我觉得还是RedFX那位阿里仁兄分析的最为透彻
12 楼 mxswl 2009-09-29  
mxswl 写道
我只知道最经典的运行时环境对内存的分块如下: 代码区 + 栈区 + 堆区.
JVM在这最基本的模型上搞出了不少花样啊.

另外我觉得字符串的引用变量(其实也就是一普通引用)的确是在栈上的,但是java里的字符串也是一个数据结构(String类),其里面真正的数据  char[] 就是放在堆区上了.

笔误,char[] 是在常量区. 不在堆上.
11 楼 mxswl 2009-09-29  
我只知道最经典的运行时环境对内存的分块如下: 代码区 + 栈区 + 堆区.
JVM在这最基本的模型上搞出了不少花样啊.

另外我觉得字符串的引用变量(其实也就是一普通引用)的确是在栈上的,但是java里的字符串也是一个数据结构(String类),其里面真正的数据  char[] 就是放在堆区上了.
10 楼 longdick 2009-09-28  
ZangXT 写道

估计所谓“所有的字符串常量(注意不是字符串对象),常量,还有对象引用都是放在栈内存中。”之说是被一个“java栈与堆”的帖子误导了。

多谢ZangXT同学指正,用你的方法做了验证,字符串常量池的确应该是存放在Perm Gen中。我差点也成了传播谬误的帮凶了呢
9 楼 ZangXT 2009-09-28  
”所有的字符串常量(注意不是字符串对象),常量,还有对象引用都是放在栈内存中。

”字符串常量、static等数据在非堆内存,也就是PermGen“
怎么二人说的不一样啊,到底谁是对的????
mumianiishiwo 写道
”所有的字符串常量(注意不是字符串对象),常量,还有对象引用都是放在栈内存中。

”字符串常量、static等数据在非堆内存,也就是PermGen“
怎么二人说的不一样啊,到底谁是对的????

估计所谓“所有的字符串常量(注意不是字符串对象),常量,还有对象引用都是放在栈内存中。”之说是被一个“java栈与堆”的帖子误导了。关于那个帖子的错误分析参考:http://zangxt.iteye.com/admin/blogs/440330
8 楼 mumianiishiwo 2009-09-28  
”所有的字符串常量(注意不是字符串对象),常量,还有对象引用都是放在栈内存中。

”字符串常量、static等数据在非堆内存,也就是PermGen“
怎么二人说的不一样啊,到底谁是对的????
7 楼 zapldy 2009-09-19  
说得比较直观,顶一个!
6 楼 energykey 2009-09-17  
不错不错,内存溢出的错误很常见的。
5 楼 downpour 2009-09-16  
好文,最好把垃圾回收的算法也一起标上,就完美了。
4 楼 plantegg 2009-09-15  
描述的不够准确:Eden空间不够的时候,直接回收,没收回的(还是好的)移到Servivor空间
3 楼 longdick 2009-09-15  
wakin2003 写道
看了您的文章,有些问题想请教一下。这个jvm内存图只说了堆内存和非堆内存(我的理解是代码存放区)。我想问栈内存,字符串,全局常量这些存储在jvm的那块内存区域内呢?麻烦您指点一下。多谢。

因为栈内存应该不属于JVM级别,是线程级别的,独立于堆内存和非堆内存(非堆内存说法不够严谨,应该为Permanent space 我文章里写的可能会引起误解,我等下会修改掉)。
我们可以用-Xss参数设置每个线程的栈内存容量。所有的字符串常量(注意不是字符串对象),常量,还有对象引用都是放在栈内存中。
总体来说一个运行中的JVM所占用内存包括=  heap size  +  perm size  +  stack size
2 楼 ZangXT 2009-09-15  
wakin2003 写道
看了您的文章,有些问题想请教一下。这个jvm内存图只说了堆内存和非堆内存(我的理解是代码存放区)。我想问栈内存,字符串,全局常量这些存储在jvm的那块内存区域内呢?麻烦您指点一下。多谢。

字符串常量、static等数据在非堆内存,也就是PermGen
1 楼 wakin2003 2009-09-15  
看了您的文章,有些问题想请教一下。这个jvm内存图只说了堆内存和非堆内存(我的理解是代码存放区)。我想问栈内存,字符串,全局常量这些存储在jvm的那块内存区域内呢?麻烦您指点一下。多谢。

相关推荐

    JVM内存模型以及垃圾回收相关资料

    JVM内存模型与垃圾回收是Java性能优化的关键部分。JVM(Java Virtual Machine)内存模型分为多个区域,包括新生代(New Generation)、老年代(Old Generation)和永久代(Permanent Generation)。新生代又细分为...

    JVM内存分配与垃圾回收详解

    JVM 内存分配与垃圾回收是 JVM 中两个非常重要的概念,本文将对 JVM 内存分配与垃圾回收进行详细的解释。 JVM 运行时数据区域 JVM 内存分配主要涉及到五个运行时数据区域:程序计数器、Java 虚拟机栈、本地方法栈...

    jvm内存基本结构及垃圾回收

    1. **堆(Heap)**:这是JVM中最大的一块内存区域,用于存储对象实例。所有通过`new`关键字创建的对象以及数组都会被分配到堆中。堆内存分为新生代(Young Generation)、老年代(Old Generation)和持久代...

    JVM入门实战/arthas实战/垃圾回收算法/垃圾回收器/jvm内存模型分析

    第三节:定位垃圾对象的依据 1.1 引用计数法 1.2 可达性算法 第四节:垃圾回收算法 1.1标记清除算法 1.2复制算法 1.3 标记整理(标记压缩)算法 第五节:垃圾回收器 1.1Serial/Serial Old收集器 1.2 ParNew收集...

    JVM垃圾回收机制

    然而,这种算法存在循环引用的问题,无法解决对象之间相互引用导致无法回收的情况,因此JVM中并不主要使用引用计数法。 2. 典型的垃圾收集算法 JVM垃圾回收机制包含以下几种典型的垃圾收集算法: - 标记-清除算法...

    JVM面试资料:JVM结构、JVM调优、四大垃圾回收算法、七大垃圾回收器

    四大垃圾回收算法:复制算法、标记-清除算法、标记-整理算法、分代收集算法 七大垃圾回收器:Serial、Serial Old、ParNew、CMS、Parallel、Parallel Old、G1 JVM调优:命令行指令,设置堆内存大小的参数

    JVM历史发展和内存回收笔记

    本文将详细探讨JVM的发展历程以及内存管理中的垃圾回收机制。 一、JVM的历史发展 1. **早期阶段**:1995年,Sun Microsystems发布了Java的第一个版本,JVM作为其核心组成部分,主要应用于嵌入式设备和网络应用。初...

    JVM垃圾回收机制与GC性能调优

    Java虚拟机(JVM)的垃圾回收(GC)机制是Java程序高效运行的关键部分,它自动管理内存,释放不再使用的对象以避免内存泄漏。本文主要探讨JVM堆内存的结构和GC的工作原理,以及如何进行性能调优。 JVM堆是Java应用...

    jvm内存模型以及垃圾回收机制.pptx

    **三、对象内存分配及垃圾回收机制** **对象内存分配**: - **栈内分配**:通过逃逸分析和标量替换技术,某些对象可以在栈上分配,提高效率并减少垃圾回收压力。 - **TLAB(Thread Local Allocation Buffer)**:每...

    JVM内存管理和垃圾回收

    1. **堆内存**:这是JVM中最大的一块内存区域,用于存储所有对象实例。当一个对象被创建时,它会分配到堆内存中。堆内存被所有线程共享,因此需要进行线程安全的管理。为了提高效率,堆内存通常被划分为新生代和老...

    JVM性能调优-JVM内存整理及GC回收.pdf

    在内存不足时,这些对象会被垃圾回收器回收。弱引用与软引用类似,也是用来描述非必须对象的,但它比软引用更弱一些,只要垃圾回收机制运行,无论当前内存是否足够,弱引用的对象都会被回收。虚引用是最弱的一种引用...

    JVM垃圾回收器和内存分配策略.zip

    垃圾回收器的主要任务是在堆内存中自动清理不再使用的对象,以避免内存泄漏。Java提供了多种不同的垃圾回收器,每种都有其特定的特性和适用场景: 1. **Serial GC**:这是最基础的垃圾回收器,适用于单线程环境。它...

    JVM内存管理和JVM垃圾回收

    Java虚拟机(JVM)内存管理和垃圾回收是Java编程中至关重要的概念,它们直接影响着程序的性能和稳定性。本文将详细解析JVM内存结构以及垃圾回收机制。 首先,JVM内存主要分为四个区域: 1. **堆(Heap)**:这是...

    JVM的工作原理及垃圾回收机制介绍

    ### JVM工作原理及垃圾回收机制详解 #### 一、JVM概述及原理 **1.1 JVM概述** Java Virtual Machine (JVM),即Java虚拟机,是一种虚构的计算机,在实际的计算机硬件上仿真模拟出的一套完整的计算机系统,用于执行...

    03-VIP-JVM内存分配机制与垃圾回收算法1

    Java虚拟机(JVM)内存分配机制和垃圾回收(Garbage Collection, GC)是Java编程中的核心概念,它们直接影响到程序的性能和稳定性。本文主要围绕JVM内存区域的分配策略,尤其是对象在新生代(Young Generation)的...

    JVM内存管理和垃圾回收知识.pdf

    压缩、不压缩和拷贝策略则是在垃圾回收速度和内存效率之间寻找平衡。 性能度量方面,关注的主要是吞吐量和暂停时间。吞吐量是应用程序总体运行时间中非垃圾回收时间的比例,而暂停时间是垃圾回收期间应用程序的暂停...

    JVM内存管理和垃圾回收.pdf

    压缩回收在垃圾回收后将存活对象移到堆的一端,形成连续空间;拷贝回收则使用两个堆,每次只在其中一个堆进行垃圾回收,并将存活对象拷贝到另一个堆,保持连续性。分代回收进一步细化了这一过程,将对象根据生存时间...

    jvm内存和垃圾回收.xmind

    自己总结的jvm中内存和垃圾回收的笔记,绘制了详细的思维导图,每个思维导图中均有详细的博文解释,方便大家学习和理解,免费分享给大家。适合jvm的爱好者和学习者

Global site tag (gtag.js) - Google Analytics