Java虚拟机GC算法GC收集器汇总

2024-06-12 10:48

本文主要是介绍Java虚拟机GC算法GC收集器汇总,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

判定对象是否还活着

Java堆中存放着几乎所有的对象实例,垃圾收集器在对堆进行回收之前,首先要确定对象是否活着,这里有几种常用的方法。

引用计数算法

给对象中添加一个引用计数器,每当一个地方引用它时,计数器+1,;当应用失效时,计数器-1;任何时刻计数器为0时,对象就是不可能再被使用的。

应用计数算法实现简单,判定效率高,在大部分情况下是一个不错的算法。但是,Java虚拟机里并没有使用该算法,其主要原因是它很难解决对象之间相互循环引用的问题。

可达性分析算法

在Java等主流实现中,都是通过可达性分析算法来判定对象是否存活的。这个算法通过一系列称为“GC Roots”的对象作为起始点,从这些节点开始向下搜索,搜索所走过的路径称为引用链,当一个对象到GC Roots没有任何引用链相连时,则证明这个对象是不可用的。

在Java语言中,GC Roots的对象包括以下几种:
  • 虚拟机栈(栈帧中的本地变量表)中引用的对象。
  • 方法区中类静态属性引用的对象。
  • 方法区中常量引用的对象。
  • 本地方法栈中JNI(即一般说的Native方法)引用的对象。

垃圾收集算法

标记-清除算法

标记-清除(Mark-Sweep)算法是最基础的收集算法,算法分为“标记”和“清除”两个阶段。首先标记出所有需要回收的对象,在标记完成后统一回收所有被标记的对象。

不足:

  • 标记和清除两个过程的效率都不高。
  • 标记清除之后会产生大量不连续的内存碎片。
复制算法

将可用内存按容量划分为大小相等的两块,每次只使用一块。当一块内存用完了,就将还存活的对象复制到另一块上面,然后再把已经使用过的内存空间一次清理掉。

复制算法实现简单,运行高效。缺点是空间使用率不高。现在的商业虚拟机都采用复制算法来回收新生代。复制算法在对象存活率较高时,会进行较多的复制操作,效率会变低。

标记-整理算法

标记整理算法,标记过程仍然与“标记-清理”算法一样,但后续步骤不是直接对可回收对象进行清理,而是让所有存活的对象都向一端移动,然后直接清理掉端边界以外的内存。标记-清理算法比较适合老年代的回收。

分代收集算法

当前商业虚拟机的垃圾收集都采用“分代收集”算法。一般是把Java堆分为新生代和老年代,在新生代中,每次垃圾收集时都发现有大批对象死去,只有少量存活,那就选用复制算法。而老年代中,因为对象存活率高、没有额外空间对它进行分配担保,就必须使用“标记-清理”算法或者“标记-整理”算法来进行回收。

垃圾收集器

如果说收集算法是内存回收的方法论,那么垃圾收集器就是内存回收的具体实现。

Serial收集器

这个收集器是单线程收集器,它在进行垃圾收集时,必须暂停其他所有的工作线程,直到它收集结束。

它的优势就是:简单而高效。

ParNew收集器

ParNew收集器其实就是Serial收集器的多线程版本,唯一的区别就是使用了多条线程进行垃圾收集。

并行(Parallel):指多条垃圾收集线程并行工作,但此时用户线程仍然处于等待状态。
并发(Concurrent):指用户线程与垃圾收集线程同时执行(但不一定是并行的,可能会交替执行),用户程序在继续运行,而垃圾收集程序运行于另一个CPU上。

Parallel Scavenge收集器

Parallel Scavenge收集器是一个新生代收集器,也是使用了复制算法,并行的多线程收集器。Parallel Scavenge的特点是,它的目标是达到一个可控制的吞吐量。吞吐量就是CPU用于运行用户代码的时间与CPU总消耗时间的比值。

Serial Old收集器

Serial Old是Serial收集器的老年代版本,它同样是一个单线程收集器,使用“标记-整理”算法。

Parallel Old收集器

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

CMS收集器

CMS(Concurrent Mark Sweep)收集器是一种以获取最短回收停顿时间为目标的收集器。CMS收集器是基于“标记-清除”算法实现的,它的运作过程相比前几种收集器来说更复杂一些,整个过程可以分为4个步骤:初始标记、并发标记、重新标记、并发清除。

其中,初始标记、重新标记这两个步骤仍然需要“Stop the World”。CMS是一款优秀的垃圾收集器,它主要的优点就是并发收集、低停顿。

CMS有3个明显的缺点:

  • CMS收集器对CPU资源非常敏感。
  • CMS收集器无法处理浮动垃圾,可能会出现“Concurrent Mode Failure”失败而导致另一次Full GC产生。(浮动垃圾:GC期间产生的垃圾,无法在当次收集中处理掉它们,只能留在下一次GC时再清理掉。)
  • CMS会产生大量空间碎片。
G1收集器

G1是一款面向服务器端应用的垃圾收集器。它具有以下几个特点:

  • 并行与并发。
  • 分代收集。
  • 空间整合。G1从整体来看是基于“标记-整理”算法实现的,从局部来看是基于“复制”算法实现的。
  • 可预测的停顿。可以让使用者指定一段时间内,消耗在垃圾收集上的时间不超过一个指定的数值。

G1之前的其他收集器进行收集的范围都是整个新生代或者老年代,而G1不再是这样。G1的Java堆布局和其他收集器有很大差异,它将整个Java堆分为多个大小相当的独立区域(Region),新生代和老年代不再是物理隔离的,它们都是一部分Region(不需要连续)的集合。

G1跟踪各个Region里面的垃圾堆积的价值大小,在后台维护一个优先列表,优先回收价值最大的Region。

G1收集器的运作大致可以分为以下几个步骤:

  • 初始标记。
  • 并发标记。
  • 最终标记。
  • 筛选回收。

G1会在初始标记、最终标记和筛选回收阶段停止用户线程的运行。

这篇关于Java虚拟机GC算法GC收集器汇总的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1054000

相关文章

Java五子棋之坐标校正

上篇针对了Java项目中的解构思维,在这篇内容中我们不妨从整体项目中拆解拿出一个非常重要的五子棋逻辑实现:坐标校正,我们如何使漫无目的鼠标点击变得有序化和可控化呢? 目录 一、从鼠标监听到获取坐标 1.MouseListener和MouseAdapter 2.mousePressed方法 二、坐标校正的具体实现方法 1.关于fillOval方法 2.坐标获取 3.坐标转换 4.坐

Spring Cloud:构建分布式系统的利器

引言 在当今的云计算和微服务架构时代,构建高效、可靠的分布式系统成为软件开发的重要任务。Spring Cloud 提供了一套完整的解决方案,帮助开发者快速构建分布式系统中的一些常见模式(例如配置管理、服务发现、断路器等)。本文将探讨 Spring Cloud 的定义、核心组件、应用场景以及未来的发展趋势。 什么是 Spring Cloud Spring Cloud 是一个基于 Spring

C++工程编译链接错误汇总VisualStudio

目录 一些小的知识点 make工具 可以使用windows下的事件查看器崩溃的地方 dumpbin工具查看dll是32位还是64位的 _MSC_VER .cc 和.cpp 【VC++目录中的包含目录】 vs 【C/C++常规中的附加包含目录】——头文件所在目录如何怎么添加,添加了以后搜索头文件就会到这些个路径下搜索了 include<> 和 include"" WinMain 和

Javascript高级程序设计(第四版)--学习记录之变量、内存

原始值与引用值 原始值:简单的数据即基础数据类型,按值访问。 引用值:由多个值构成的对象即复杂数据类型,按引用访问。 动态属性 对于引用值而言,可以随时添加、修改和删除其属性和方法。 let person = new Object();person.name = 'Jason';person.age = 42;console.log(person.name,person.age);//'J

java8的新特性之一(Java Lambda表达式)

1:Java8的新特性 Lambda 表达式: 允许以更简洁的方式表示匿名函数(或称为闭包)。可以将Lambda表达式作为参数传递给方法或赋值给函数式接口类型的变量。 Stream API: 提供了一种处理集合数据的流式处理方式,支持函数式编程风格。 允许以声明性方式处理数据集合(如List、Set等)。提供了一系列操作,如map、filter、reduce等,以支持复杂的查询和转

Java面试八股之怎么通过Java程序判断JVM是32位还是64位

怎么通过Java程序判断JVM是32位还是64位 可以通过Java程序内部检查系统属性来判断当前运行的JVM是32位还是64位。以下是一个简单的方法: public class JvmBitCheck {public static void main(String[] args) {String arch = System.getProperty("os.arch");String dataM

详细分析Springmvc中的@ModelAttribute基本知识(附Demo)

目录 前言1. 注解用法1.1 方法参数1.2 方法1.3 类 2. 注解场景2.1 表单参数2.2 AJAX请求2.3 文件上传 3. 实战4. 总结 前言 将请求参数绑定到模型对象上,或者在请求处理之前添加模型属性 可以在方法参数、方法或者类上使用 一般适用这几种场景: 表单处理:通过 @ModelAttribute 将表单数据绑定到模型对象上预处理逻辑:在请求处理之前

eclipse运行springboot项目,找不到主类

解决办法尝试了很多种,下载sts压缩包行不通。最后解决办法如图: help--->Eclipse Marketplace--->Popular--->找到Spring Tools 3---->Installed。

JAVA读取MongoDB中的二进制图片并显示在页面上

1:Jsp页面: <td><img src="${ctx}/mongoImg/show"></td> 2:xml配置: <?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001

Java面试题:通过实例说明内连接、左外连接和右外连接的区别

在 SQL 中,连接(JOIN)用于在多个表之间组合行。最常用的连接类型是内连接(INNER JOIN)、左外连接(LEFT OUTER JOIN)和右外连接(RIGHT OUTER JOIN)。它们的主要区别在于它们如何处理表之间的匹配和不匹配行。下面是每种连接的详细说明和示例。 表示例 假设有两个表:Customers 和 Orders。 Customers CustomerIDCus