JVM - 再聊GC垃圾收集算法及垃圾收集器

简介: JVM - 再聊GC垃圾收集算法及垃圾收集器

20200629220908526.png


Pre


JVM-04垃圾收集Garbage Collection(上)【垃圾对象的判定】

JVM-05垃圾收集Garbage Collection(中)【垃圾收集算法】

JVM-06垃圾收集Garbage Collection(下)【垃圾收集器】


分代收集理论


当前虚拟机的垃圾收集都采用分代收集算法 , 意思就是根据对象存活周期的不同将 java堆分为新生代和老年代,这样就可以根据各个年代的特点选择合适的垃圾收集算法。


举个例子,新生代中,大部分对象都是朝生夕死的,所以可以选择复制算法, 只复制那些少量存活的对象。


而老年代,绝大部分都是一些顽固的对象,存货周期较长,如果选择复制算法,就要复制很多对象,效率自然不高 ,所以得换个算法,通常会选择标记-清除”或“标记-整理”算法进行垃圾收集。


常见的垃圾收集算法


20200629221604292.png

标记-清除算法

两个阶段 “标记” + “清除”

  1. 标记存活的对象, 也可以反过来,标记出所有需要回收的对象
  2. 标记完成后,统一回收所有未被标记的对象(一般选择这种)



20200629222942862.png


存在两个问题

  1. 如果需要标记的对象太多,效率不高 【效率问题】
  2. 标记清除后会产生大量不连续的碎片 【内存碎片】

基于这个缺点,进化除了 标记-整理 ,多了一步整理内存碎片。


标记-复制算法


它将内存分为大小相同的两块,每次使用其中的一块。

当这一块的内存使用完后,就将还存活的对象复制到另一块去,然后再把使用的空间一次清理掉。

这样就使每次的内存回收都是对内存区间的一半进行回收。

20200629222436784.png


  • 优点:常用于新生代,存活的对象少,复制这些存活的对象,效率高。
  • 缺点 显而易见: 浪费空间

标记-整理算法


根据老年代的特点演化出一种标记算法, 标记过程同“标记-清除” 。

移动所有存活的对象,且按照内存地址次序依次排列,然后将末端内存地址以后的内存全部回收。因此,第二阶段才称为整理阶段。


20180729171526874.jpg


垃圾收集器


20200629225619675.png


为什么会有这么多GC 回收器?


因为目前没有任何一款是十分完美的,我们需要做的就是充分了解这些GC Collector 根据业务特点选择合适的GC Collector 。


Serial收集器


两个收集器 。一个 新生代Serial , 一个 老年代 Serial Old .

新生代采用复制算法,老年代采用标记-整理算法。


JVM参数

-XX:+UseSerialGC (新生代)  -XX:+UseSerialOldGC (老年代)

Serial(串行)收集器是最基本、历史最悠久的垃圾收集器 。


20200629230423264.png


字面意思:

  1. 单线程GC
  2. GC的时候STW
  3. 新生代 Serial , 复制算法
  4. 老年代Serial Old , 标记整理算法

Serial Old收集器是Serial收集器的老年代版本,也是一个单线程收集器。


它主要有两大用途:

  • 一种用途是在JDK1.5以及以前的版本中与Parallel Scavenge收集器搭配使用,
  • 另一种用途是作为CMS收集器的后备方案


Parallel Scavenge收集器 【JDK8默认】

Serial收集器的多线程版本

JVM参数

-XX:+UseParallelGC(年轻代),-XX:+UseParallelOldGC(老年代)

20200629231552988.png



默认的收集线程数跟cpu核数相同,当然也可以用参数(-XX:ParallelGCThreads)指定收集线程数,但是一般不推荐修改。


Parallel Scavenge收集器关注点是吞吐量(高效率的利用CPU)


CMS等垃圾收集器的关注点更多的是用户线程的停顿时间(提高用户体验)


新生代采用复制算法,老年代采用标记-整理算法。


Parallel Old收集器是Parallel Scavenge收集器的老年代版本。使用多线程和“标记-整理”算法。


在注重吞吐量以及CPU资源的场合,都可以优先考虑 Parallel Scavenge收集器和Parallel Old收集器(JDK8默认的新生代和老年代收集器)。


ParNew收集器


What ? ParNew ? 不是有个Parallel 吗? 又出来这个主要是因为Parallel 不能和CMS搭配使用,所以官方又弄出个这么个玩意儿。

JVM参数

-XX:+UseParNewGC


20200629232739724.png


ParNew收集器其实跟Parallel收集器很类似,区别主要在于它可以和CMS收集器配合使用。

新生代采用复制算法,老年代采用标记-整理算法。

互联网公司 小内存的机器 主流的GC组合 就是 ParNew + CMS .


CMS 【重点掌握】


JVM参数

-XX:+UseConcMarkSweepGC(老年代)

20200630001210148.png


详细的四个步骤、CMS存在的问题以及三色标记我们下篇博文继续展开

相关文章
|
10天前
|
算法 网络协议 Java
【JVM】——GC垃圾回收机制(图解通俗易懂)
GC垃圾回收,标识出垃圾(计数机制、可达性分析)内存释放机制(标记清除、复制算法、标记整理、分代回收)
|
3月前
|
缓存 算法 Java
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
这篇文章详细介绍了Java虚拟机(JVM)中的垃圾回收机制,包括垃圾的定义、垃圾回收算法、堆内存的逻辑分区、对象的内存分配和回收过程,以及不同垃圾回收器的工作原理和参数设置。
110 4
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
|
2月前
|
存储 监控 Java
JVM进阶调优系列(8)如何手把手,逐行教她看懂GC日志?| IT男的专属浪漫
本文介绍了如何通过JVM参数打印GC日志,并通过示例代码展示了频繁YGC和FGC的场景。文章首先讲解了常见的GC日志参数,如`-XX:+PrintGCDetails`、`-XX:+PrintGCDateStamps`等,然后通过具体的JVM参数和代码示例,模拟了不同内存分配情况下的GC行为。最后,详细解析了GC日志的内容,帮助读者理解GC的执行过程和GC处理机制。
|
2月前
|
算法 Java
JVM有哪些垃圾回收算法?
(1)标记清除算法: 标记不需要回收的对象,然后清除没有标记的对象,会造成许多内存碎片。 (2)复制算法: 将内存分为两块,只使用一块,进行垃圾回收时,先将存活的对象复制到另一块区域,然后清空之前的区域。用在新生代 (3)标记整理算法: 与标记清除算法类似,但是在标记之后,将存活对象向一端移动,然后清除边界外的垃圾对象。用在老年代
23 0
|
3月前
|
存储 监控 算法
JVM调优深度剖析:内存模型、垃圾收集、工具与实战
【10月更文挑战第9天】在Java开发领域,Java虚拟机(JVM)的性能调优是构建高性能、高并发系统不可或缺的一部分。作为一名资深架构师,深入理解JVM的内存模型、垃圾收集机制、调优工具及其实现原理,对于提升系统的整体性能和稳定性至关重要。本文将深入探讨这些内容,并提供针对单机几十万并发系统的JVM调优策略和Java代码示例。
68 2
|
3月前
|
Arthas 监控 Java
JVM知识体系学习七:了解JVM常用命令行参数、GC日志详解、调优三大方面(JVM规划和预调优、优化JVM环境、JVM运行出现的各种问题)、Arthas
这篇文章全面介绍了JVM的命令行参数、GC日志分析以及性能调优的各个方面,包括监控工具使用和实际案例分析。
88 3
|
3月前
|
算法 Java
JVM进阶调优系列(4)年轻代和老年代采用什么GC算法回收?
本文详细介绍了JVM中的GC算法,包括年轻代的复制算法和老年代的标记-整理算法。复制算法适用于年轻代,因其高效且能避免内存碎片;标记-整理算法则用于老年代,虽然效率较低,但能有效解决内存碎片问题。文章还解释了这两种算法的具体过程及其优缺点,并简要提及了其他GC算法。
 JVM进阶调优系列(4)年轻代和老年代采用什么GC算法回收?
|
3月前
|
存储 算法 Java
【JVM】垃圾释放方式:标记-清除、复制算法、标记-整理、分代回收
【JVM】垃圾释放方式:标记-清除、复制算法、标记-整理、分代回收
75 2
|
3月前
|
缓存 算法 Java
GC垃圾收集算法
这篇文章详细讨论了垃圾收集(GC)的几种算法,包括引用计数、可达性分析、标记-清除、标记-复制和标记-整理算法,并介绍了这些算法的优缺点和适用场景。
43 0
GC垃圾收集算法
|
3月前
|
存储 Java PHP
【JVM】垃圾回收机制(GC)之引用计数和可达性分析
【JVM】垃圾回收机制(GC)之引用计数和可达性分析
88 0