浅谈JVM垃圾回收有哪些常用算法

 更新时间:2021年06月22日 11:47:50   作者:亭中独酌  
今天给大家带来的是关于Java虚拟机的相关知识,文章围绕着JVM垃圾回收有哪些常用算法展开,文中有非常详细的介绍及代码示例,需要的朋友可以参考下

一、前言:

垃圾回收:

在未来的JDK中可能G1会为ZGC所取代

先问自己几个问题:

什么是垃圾?

  • 垃圾就是堆内存中(范指)没有任何指针指向的对象实体。不具有可达性。

为什么要回收垃圾?

  • 因为我们的内存是有限的,内存长时间不清理就会导致内存溢出,OOM;
  • 只要是程序正在跑,那么就不断生成新的对象,我们需要GC开辟新的空间分配给新的对象

我们怎么回收垃圾?

  • 依靠Java的自动内存回收机制,机制的优劣由算法决定;
  • 或者说是机制的适配度由算法和应用场景共同决定。

什么时候回收垃圾?

  • 当堆中的实体对象没有任何指针指向的时候

二、GC的标记阶段算法:

标记&清除

1、引用计数(Reference Counting):

Java已经摈弃了这种算法,因为此算法需要的额外处理过多

【优】效率高,python也在用,就像论文的引用因子一样,没有用的文章就应该多多回收,清理学术垃圾。

【缺】无法处理对象的相互“循环引用”,一旦形成了引用环,就没有办法去解决。进而造成内存泄漏。

2、可达性分析⭐(根搜索、Tracing Garage Collection):

GC Roots = 起始节点集,从GC Roots开始向下搜索,连接的路径为引用链,GC Roots不可达的对象被判为不可用。

哪些是GC Roots?

  • 虚拟栈上的栈帧的局部变量表引用的对象;
  • 方法区上常量引用
  • 方法区上静态变量
  • 被同步锁修饰的对象
  • 除了堆区,和堆有联系的都是起始节点……

【优】解决了循环引用的缺点

【缺】需要遍历

三、垃圾收集算法:

标记清除算法
复制算法
标记清除整理算法

标记-清除算法:

先mark可达对象,从根节点开始进行线性遍历

【优】够平均

【缺】效率不高,GC的时候导致STW,清楚后存在内存碎片(会存在一个空闲列表)

这是最快的清除算法

复制算法

先把空间分为两个部分,把标记的对象规整地移到另一个空间中(指针碰撞的方式)

【优】高效,无需mark/sweep;没有内存碎片;

【缺】牺牲了大量的空间,”最好你们全部是垃圾!“

标记-清除-整理算法

在标记之后清除完了再进行整理,属于标记清除算法的优化版,无空闲列表

【优】无空闲列表,无内存碎片;空间开销低

【缺】时间慢,需要进行多次操作。

四、finalize&内存分析工具

finalization——免死金牌

finalize是给GC调用的

【问】回收的时候会涉及到哪些操作?会伴随着什么状态?

  • 可触及:正常状态,在GC Roots的引用链上;
  • 可复活:需要重写finalize方法才有的,“皇帝赐给你的重写finalize方法”
  • 不可触及:finalize免死金牌只能用一次,如果没有重写的finalize方法,那么就直接挂了。

MAT & GC Roots:

Memory Analyzer Tools 内存分析工具

分析dump文件:根据GC Roots去溯源,监控内存泄漏→ JProfiler

分区算法

将堆空间分成小空间是为了降低停顿时间,降低延迟

实际的使用都是复合算法。

String

final是写死的,不能继承也不能做任何修改;

Serializable修饰是跨进程

Comparable可比较的

到此这篇关于浅谈JVM垃圾回收有哪些常用算法的文章就介绍到这了,更多相关JVM垃圾回收算法内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java枚举之EnumSet详解

    Java枚举之EnumSet详解

    这篇文章主要介绍了Java枚举之EnumSet详解,使用时进行与或运算,但是定义多了之后,会很乱、臃肿,编写容易出错,EnumSet可以实现类似的功能,且使用起来很简洁,需要的朋友可以参考下
    2023-12-12
  • Java设计模式之责任链模式简介

    Java设计模式之责任链模式简介

    这篇文章主要介绍了Java设计模式之责任链模式,需要的朋友可以参考下
    2014-07-07
  • JUnit4 Hamcrest匹配器常用方法总结

    JUnit4 Hamcrest匹配器常用方法总结

    这篇文章主要介绍了JUnit4 Hamcrest匹配器常用方法总结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-11-11
  • Java中的包(package)是什么和使用方法

    Java中的包(package)是什么和使用方法

    包是Java中一种强大的组织代码的工具,它们帮助开发者将代码分组,防止命名冲突,并通过控制访问级别来增强代码的安全性,这篇文章主要介绍了Java中的包(package)是什么和如何使用它们,需要的朋友可以参考下
    2024-07-07
  • Spring boot 启动流程及外部化配置方法

    Spring boot 启动流程及外部化配置方法

    平时我们开发Spring boot 项目的时候,一个SpringBootApplication注解加一个main方法就可以启动服务器运行起来,那它到底是怎么运行起来的呢?这篇文章主要介绍了Spring boot 启动流程及外部化配置,需要的朋友可以参考下
    2022-12-12
  • Java实现Token登录验证的项目实践

    Java实现Token登录验证的项目实践

    本文主要介绍了Java实现Token登录验证的项目实践,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-03-03
  • springboot下配置多数据源的方法

    springboot下配置多数据源的方法

    本篇文章主要介绍了springboot下配置多数据源的方法,具有一定的参考价值,有兴趣的可以了解一下
    2017-04-04
  • Java8中AbstractExecutorService与FutureTask源码详解

    Java8中AbstractExecutorService与FutureTask源码详解

    这篇文章主要给大家介绍了关于Java8中AbstractExecutorService与FutureTask的相关资料,文中通过实例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2022-01-01
  • 详解eclipse将项目打包成jar文件的两种方法及问题解决方法

    详解eclipse将项目打包成jar文件的两种方法及问题解决方法

    本文给大家介绍了eclipse中将项目打包成jar文件的两种方法及其遇到问题解决方法,本文图文并茂给大家介绍的非常详细,需要的朋友可以参考下
    2017-12-12
  • Java使用Scanner类进行控制台输入实现方法

    Java使用Scanner类进行控制台输入实现方法

    这篇文章主要介绍了Java使用Scanner类进行控制台输入实现方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-12-12

最新评论