通过图文给你讲明白java GC的垃圾回收机制

简介: 通过图文给你讲明白java GC的垃圾回收机制

本文原链接

http://www.oracle.com/webfolder/technetwork/tutorials/obe/java/gc01/index.html

1. JAVA GC 概述



JAVA GC采用了分代思想,将java堆分成新生代,年老代,永久代。GC算法主要有标记-清除,标记-压缩,复制算法。


  • 新生代:新生代被分成三个部分 eden区和2个survivor区(from和to两个分区)。当创建对象,需要jvm分配内存时,会在新生代的eden区寻找合适的内存区域。如果当eden区内存不够时,会触发minor GC。eden区存活对象和from区的存活对象将会被复制到to区。当to区的对象年龄超过了晋升的年龄设置,对象将被提升到老年代。新生代GC用的是复制算法
  • 年老代: 年老代里存放的都是存活时间较久的,大小较大的对象,因此年老代使用标记整理算法。当年老代容量满的时候,会触发一次Major GC(full GC),回收年老代和年轻代中不再被使用的对象资源。年老代算法用的是标记-清除
  • 永久代:指内存的永久保存区域,主要存放Class和Meta(元数据)的信息。Class在被加载的时候被放入永久区域。它和和存放实例的区域不同,GC不会在主程序运行期对永久区域进行清理。所以这也导致了永久代的区域会随着加载的Class的增多而胀满,最终抛出OOM异常。


2. 图文描述标记-清除和标记-压缩



Step 1:标记(Marking)

GC的第一步叫做标记。在这个步骤GC通过遍历内存区辨别哪些内存在使用,哪些内容没有使用。并做好标记803fce8c420392d85533b88fe5686043.png


如上图蓝色的表示存活的对象,金黄色表示垃圾对象。在标记阶段,需要扫描整个该内存区的对象,并标记。这个过程可能会比较耗时


Step 2: 清除(Normal Deletion)


清除阶段移除掉垃圾对象,并且用一个链表维护空闲的区域


[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-c1T7lU42-1606210937650)(http://www.oracle.com/webfolder/technetwork/tutorials/obe/java/gc01/images/gcslides/Slide1b.png)]


内存分配器持有空闲内存区的引用,以便分配内存给新的对象


Step 2a: 压缩(Deletion with Compacting)


为了提升性能,在Step 2的基础上,在删除完垃圾对象后。我们可以把存活的对象移动到内存区的头部。这样下次分配内存的时候会更快。主要原因是标记-清除会造成比较大的内存碎片,每当需要分配内存时,都需要遍历空闲链表。而压缩算法,会把内存碎片整理成一个大的完整内存块。

cd1a0df4367b1bf13d1a47d668c09772.png

3. 分代垃圾回收



为什么要采用分代垃圾回收?


正如前面所说,标记和压缩对象,对java虚拟机而言会比较耗时。当java虚拟机分配了越来越多的对象后。GC所花费的时间将会更长。然而根据经验分析,绝大多数的对象存活时间都比较短。这样我们可以把存活长的对象和存活短的对象隔离开。这样GC会更加高效

37542ad68f4f85892fa9b98a8569ebf9.gif

JVM分代


将jvm堆内存分割成更小的内存区,会提高jvm的gc性能。堆被分成 (新生代)Young Generation,(年老代)Old or Tenured Generation, and (永久代)Permanent Generation


6c8a57cc4bad5a02e9e7fcc3d128caf9.png

  • 所有的对象都会在新生代中分配内存。当新生代内存不够的时候。将会出发minor GC。如果新生代中的对象存活时间都很短,呢么minor GC的效率将会很高。如果新生代里面充满了垃圾对象,那么回收速度将会很快(因为标记的时间短了)。一些存活下来的对象年龄将会增加,并且最终会被移动到年老代
  • Stop the World Event:所有的minor GC都是"Stop the World"事件。这意味着出来GC线程,程序的其他线程对将暂停知道GC完成。Minor GC都是Stop the World Event
  • 年老代是用来存储长存活时间的对象。典型的我们可以给对象设定一个年龄界限,当新生代的对象存活年龄超过这个界限,对象将会从新生代移动到年老代。当年老代的内存不够的时候,将会出发Major GC。Major GC也是Stop the World Event。通常来说Major GC比Minor GC更慢,因为Major GC回收的是整个新生代和年老代的所有垃圾对象。因此对于响应性高的程序,应该尽量减少Major GC。需要注意的是Stop the World Event的时间受到在年老代中使用的垃圾回收器的影响,不受新生代的影响
  • 永久代包含了JVM的元数据。包括类信息,方法信息等。永久代由JVM在应用运行期生成。另外 javase的类库中的类信息也可能存放在这里


分代垃圾回收的处理过程



现在你明白为什么堆分成不同的世代,现在是时候看看这些空间是如何相互作用的。 下面的图片将介绍JVM中的对象分配和老化过程。


1. 一开始,任何新的对象都会在eden空间分配内存,两个survivor空间一开始都是空的

ee3cd7ea17f1b9caf5293d83471ec0c2.png

2. 当eden空间被填满了,minor GC将被触发

9089abaee5c96db491a19a5da8750b23.png


S0 survivor区对象里的 1 3表示对象的年龄


3. Eden空间的存活的对象将被复制到第一个survivor空间,年龄+1,垃圾对象将会被清除掉


[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-GIW40p7p-1606210937656)

(http://www.oracle.com/webfolder/technetwork/tutorials/obe/java/gc01/images/gcslides/Slide6.png)]


4. 下一次minor GC发生时,Eden空间的存活对象将被复制到空闲的survivor空间S1(年龄+1),另外在前一次minor GC S0空间的存活对象也会被复制到S1(年龄+1),垃圾对象会被清除掉

08c0b53b6da12079a3928b073171926d.png

5. 下一次minor GC发生时,还是重复第4条的内容,只是两个survivor空间对调了,这次是从S1复制到S0空间


[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-acfW5s1W-1606210937657)

(http://www.oracle.com/webfolder/technetwork/tutorials/obe/java/gc01/images/gcslides/Slide9.png)]


6. 新生代晋升到年老代。当minor GC发生时,如果survivor空间中的对象年龄超过了晋升的年龄限定,对象会被复制到年老代

40f3dc755d9d01b11edfdbb3be734259.png

7. 当minor GC不断触发,将会有对象不断被晋升到年老代

615f9cad0f698393531edf09d9c6de32.png8. 上面的图文完美的解释了minor GC的处理过程。最终,当年老代的内存被填满的时候,将会触发major GC。Major GC在年老代用的是标记压缩算法。同时新生代的对象将被清除

d6fccf9a5f9c4f5885b41c9d57ddef71.png

4.后续


后面我将计划写一系列关于垃圾回收的博文。主要内容会涉及到垃圾回收的算法描述。

相关文章
|
2月前
|
监控 算法 Java
Java虚拟机(JVM)的垃圾回收机制深度解析####
本文深入探讨了Java虚拟机(JVM)的垃圾回收机制,旨在揭示其背后的工作原理与优化策略。我们将从垃圾回收的基本概念入手,逐步剖析标记-清除、复制算法、标记-整理等主流垃圾回收算法的原理与实现细节。通过对比不同算法的优缺点及适用场景,为开发者提供优化Java应用性能与内存管理的实践指南。 ####
|
1月前
|
监控 算法 Java
Java虚拟机(JVM)垃圾回收机制深度剖析与优化策略####
本文作为一篇技术性文章,深入探讨了Java虚拟机(JVM)中垃圾回收的工作原理,详细分析了标记-清除、复制算法、标记-压缩及分代收集等主流垃圾回收算法的特点和适用场景。通过实际案例,展示了不同GC(Garbage Collector)算法在应用中的表现差异,并针对大型应用提出了一系列优化策略,包括选择合适的GC算法、调整堆内存大小、并行与并发GC调优等,旨在帮助开发者更好地理解和优化Java应用的性能。 ####
56 0
|
25天前
|
算法 网络协议 Java
【JVM】——GC垃圾回收机制(图解通俗易懂)
GC垃圾回收,标识出垃圾(计数机制、可达性分析)内存释放机制(标记清除、复制算法、标记整理、分代回收)
|
1月前
|
存储 监控 算法
Java虚拟机(JVM)垃圾回收机制深度解析与优化策略####
本文旨在深入探讨Java虚拟机(JVM)的垃圾回收机制,揭示其工作原理、常见算法及参数调优方法。通过剖析垃圾回收的生命周期、内存区域划分以及GC日志分析,为开发者提供一套实用的JVM垃圾回收优化指南,助力提升Java应用的性能与稳定性。 ####
|
2月前
|
机器学习/深度学习 监控 算法
Java虚拟机(JVM)的垃圾回收机制深度剖析####
本文深入探讨Java虚拟机(JVM)的垃圾回收机制,揭示其工作原理、常见算法、性能调优策略及未来趋势。通过实例解析,为开发者提供优化Java应用性能的思路与方法。 ####
63 1
|
2月前
|
监控 算法 Java
Java虚拟机垃圾回收机制深度剖析与优化策略####
【10月更文挑战第21天】 本文旨在深入探讨Java虚拟机(JVM)中的垃圾回收机制,揭示其工作原理、常见算法及参数调优技巧。通过案例分析,展示如何根据应用特性调整GC策略,以提升Java应用的性能和稳定性,为开发者提供实战中的优化指南。 ####
49 5
|
2月前
|
算法 Java 开发者
Java内存管理与垃圾回收机制深度剖析####
本文深入探讨了Java虚拟机(JVM)的内存管理机制,特别是其垃圾回收机制的工作原理、算法及实践优化策略。不同于传统的摘要概述,本文将以一个虚拟的“城市环卫系统”为比喻,生动形象地揭示Java内存管理的奥秘,旨在帮助开发者更好地理解并调优Java应用的性能。 ####
|
2月前
|
缓存 算法 Java
本文聚焦于Java内存管理与调优,介绍Java内存模型、内存泄漏检测与预防、高效字符串拼接、数据结构优化及垃圾回收机制
在现代软件开发中,性能优化至关重要。本文聚焦于Java内存管理与调优,介绍Java内存模型、内存泄漏检测与预防、高效字符串拼接、数据结构优化及垃圾回收机制。通过调整垃圾回收器参数、优化堆大小与布局、使用对象池和缓存技术,开发者可显著提升应用性能和稳定性。
62 6
|
1月前
|
存储 监控 算法
Java内存管理的艺术:深入理解垃圾回收机制####
本文将引领读者探索Java虚拟机(JVM)中垃圾回收的奥秘,解析其背后的算法原理,通过实例揭示调优策略,旨在提升Java开发者对内存管理能力的认知,优化应用程序性能。 ####
53 0
|
2月前
|
监控 算法 Java
深入理解Java的垃圾回收机制
【10月更文挑战第22天】在Java的世界里,有一个默默无闻却至关重要的角色——垃圾回收(Garbage Collection, GC)。就像城市的清洁工一样,它默默地清理着不再使用的内存空间,确保我们的程序运行得既高效又稳定。但你真的了解垃圾回收是如何工作的吗?让我们一起探索这个看似简单却充满奥秘的过程,看看它是如何影响你的Java应用性能的。