JVM学习之六 垃圾收集算法

2024-04-18 03:18

本文主要是介绍JVM学习之六 垃圾收集算法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

垃圾收集算法

    因为JVM的内存分为不同的区域,作用也不相同,存储的元素生存周期也不同,这就决定了垃圾收集算法有着不相同的多种算法。目前的收集算法有标记-清除算法,复制算法和标记整理算法,虽然也有分代收集算法的叫法,但其实是在不同的内存区域根据对象的生存特点使用以上某种收集算法而已,故不认为是一种独立的收集算法。现将以上三种收集算法记录如下:

标记-清除算法    

    标记清除(Mark-Sweep)算法是最基础的算法,其他算法是对该算法的改进。标记清除算法分为两个阶段,分别是标记阶段和清除阶段。

    标记阶段就是之前博客(四种引用和垃圾回收的两次标记)中提到的两次标记,用来找出那些对象是准备要回收的。清除阶段就是就是对要回收的对象进行清除,回收那些空间。

    该算法的主要缺点是效率不高和空间散碎。效率不高指的是标记和清除两个过程的效率都不高,而空间散碎只的是执行了垃圾回收的内存会产生很多大量不连续的内存碎片,这在创建需要较大连续内存的对象比如大数组时,不得不触发一次GC。老年代有可能会采用该种算法,比如CMS收集器,也就是concurrent makr sweep 收集器。

复制算法

    复制算法针对标记-清除算法改进了效率问题。它将内存按照容量划分为大小相等的两块,每次只使用其中的一块,当这一块使用完了就将存活对象复制到另一块内存中,然后将其全部回收。这种算法的优点是不用考虑内存碎片等复杂情况,按顺序分配内存即可,简单高效。缺点就是可用的最大内容空间缩小了一半,代价未免太高了一点。

    现在的商业虚拟机都采用这种算法回收新生代。因为新生代的对象绝大多数都是朝生夕死的(IBM公司研究表明98%),所以为了更高效的利用内存空间,将新生代中划分为一块较大内存的Eden区和两块较小内存的Survivor区,每次同时使用Eden区和其中的一个Survivor区,当回收时,将Eden区和使用的Survivor区的存活对象一次性的复制到另外一块Survivor区,最后清理掉Eden区和刚才使用的Survivor区。HotSpot虚拟机默认的Eden区和Survivor区大小比例是8:1。HotSpot虚拟机是目前使用最广泛的虚拟机,是SunJDK和OpenJDK所带的虚拟机。

    内存担保。当当前未使用的Survivor区的内存不能保证将Eden区和另外正在使用的Survivor区内存中的存活对象都复制到其中时,我们需要将其复制到老年代,这叫做内存担保,以免新生代中出现大量非朝生夕死的对象的场景时,复制存活对象失败。

    例如Serial收集器,ParNew收集器,Parallel Scavenge收集器以及G1收集器中从局部看两个region之间都是使用该算法。

标记-整理算法

    复制算法在存活对象时间较长和较多时,会导致效率降低;而且内存分配担保会降低内存的使用效率,不适合在对象存活时间长的场景下使用,因此提出了标记整理算法。该算法也是分两个阶段,标记阶段与整理阶段,其中标记极端与标记-清除算法相同,整理阶段是让存活对象向一端移动,然后清理掉存活对象边界意外的内存,也就是将存活对象和非存活对象整理出来,非存活对象的内存放在一起处理。所以该算法也是适用于老年代的。

    像Serial Old收集器,Parallel Old收集器,以及G1收集器从整体看都是使用该算法的。

    垃圾收集算法是垃圾收集的方法论,真正实现垃圾回收功能的是垃圾收集器,只有充分理解每部分内存中对象的生命周期和每部分内存的回收算法,我们才能知道这部分内存使用哪个收集器去收集。

声明:本博客参考了周志明《深入理解Java虚拟机》2版    

    

这篇关于JVM学习之六 垃圾收集算法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Spring Cloud LoadBalancer 负载均衡详解

《SpringCloudLoadBalancer负载均衡详解》本文介绍了如何在SpringCloud中使用SpringCloudLoadBalancer实现客户端负载均衡,并详细讲解了轮询策略和... 目录1. 在 idea 上运行多个服务2. 问题引入3. 负载均衡4. Spring Cloud Load

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

在 Spring Boot 中使用 @Autowired和 @Bean注解的示例详解

《在SpringBoot中使用@Autowired和@Bean注解的示例详解》本文通过一个示例演示了如何在SpringBoot中使用@Autowired和@Bean注解进行依赖注入和Bean... 目录在 Spring Boot 中使用 @Autowired 和 @Bean 注解示例背景1. 定义 Stud

如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别详解

《如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别详解》:本文主要介绍如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别的相关资料,描述了如何使用海康威视设备网络SD... 目录前言开发流程问题和解决方案dll库加载不到的问题老旧版本sdk不兼容的问题关键实现流程总结前言作为

SpringBoot中使用 ThreadLocal 进行多线程上下文管理及注意事项小结

《SpringBoot中使用ThreadLocal进行多线程上下文管理及注意事项小结》本文详细介绍了ThreadLocal的原理、使用场景和示例代码,并在SpringBoot中使用ThreadLo... 目录前言技术积累1.什么是 ThreadLocal2. ThreadLocal 的原理2.1 线程隔离2

springboot将lib和jar分离的操作方法

《springboot将lib和jar分离的操作方法》本文介绍了如何通过优化pom.xml配置来减小SpringBoot项目的jar包大小,主要通过使用spring-boot-maven-plugin... 遇到一个问题,就是每次maven package或者maven install后target中的ja

Java中八大包装类举例详解(通俗易懂)

《Java中八大包装类举例详解(通俗易懂)》:本文主要介绍Java中的包装类,包括它们的作用、特点、用途以及如何进行装箱和拆箱,包装类还提供了许多实用方法,如转换、获取基本类型值、比较和类型检测,... 目录一、包装类(Wrapper Class)1、简要介绍2、包装类特点3、包装类用途二、装箱和拆箱1、装

如何利用Java获取当天的开始和结束时间

《如何利用Java获取当天的开始和结束时间》:本文主要介绍如何使用Java8的LocalDate和LocalDateTime类获取指定日期的开始和结束时间,展示了如何通过这些类进行日期和时间的处... 目录前言1. Java日期时间API概述2. 获取当天的开始和结束时间代码解析运行结果3. 总结前言在J

Java深度学习库DJL实现Python的NumPy方式

《Java深度学习库DJL实现Python的NumPy方式》本文介绍了DJL库的背景和基本功能,包括NDArray的创建、数学运算、数据获取和设置等,同时,还展示了如何使用NDArray进行数据预处理... 目录1 NDArray 的背景介绍1.1 架构2 JavaDJL使用2.1 安装DJL2.2 基本操

最长公共子序列问题的深度分析与Java实现方式

《最长公共子序列问题的深度分析与Java实现方式》本文详细介绍了最长公共子序列(LCS)问题,包括其概念、暴力解法、动态规划解法,并提供了Java代码实现,暴力解法虽然简单,但在大数据处理中效率较低,... 目录最长公共子序列问题概述问题理解与示例分析暴力解法思路与示例代码动态规划解法DP 表的构建与意义动