JVM自动内存管理之垃圾收集算法

简介: 文章概述了JVM内存管理和垃圾收集的基本概念,提供一个关于JVM内存管理和垃圾收集的基础理解框架。

前一篇已经了解过JVM有哪些运行时数据区域,以及每个区域的作用即存储了哪些数据,本篇文章将了解JVM对这块区域内存的回收方案。

因为机器内存是有限制的,不可能让程序一直运行并不停分配内存,而不对无需再使用的内存进行回收管理再利用,因此内存的回收管理是很重要的

内存回收管理即垃圾收集工作的正常进行一定要完成下面3个工作

第一、哪些内存是需要回收的?

第二、什么时候回收呢?

第三、怎么回收这些需要被回收的内存?

第一个问题,哪些内存需要回收?让我们回顾下JVM运行时数据区来了解下。

程序计数器: 这块区域是线程私有的,线程创建而创建,线程消亡而消亡,并且这块区域占有空间足够小不需要考虑动态回收问题

本地方法栈:这块区域是线程私有的,线程创建而创建,线程消亡而消亡,栈中的栈帧随着方法进入和退出,方法结束内存就可以自动回收了

JAVA虚拟机栈:这块区域是线程私有的,线程创建而创建,线程消亡而消亡,栈中的栈帧随着方法进入和退出,方法结束内存就可以自动回收了

堆:该区域存储的是对象大部分是程序运行过程中动态产生的,许多对象使用后可以回收掉,所以回收内存主要关注这块区域

方法区:方法区有些常量以及一些无用的类,也是需要动态回收的

第二个问题,什么时候回收呢?

当对象不再需要使用时就可以回收了,那么回收对象前肯定需要确定内存中哪些对象还"活着",哪些对象已经"死去"。

判断对象已经|死去的方法有一下几种:

1.引用计数法

这种方法实现比较简单,给每个对象都设置一个计数器,当有对象引用它时,计数器+1,当引用失效时,计数器减1

这种方法很难解决循环引用的问题,即A引用B,B又引用A的情况

2.可达性分析算法

这种办法通过将一系列的GC Roots的对象作为起始点,从这些节点开始向下搜搜,搜索所走过的路称为引用链,当一个对象到GC Roots没有热为奴引用链相连,则证明此对象是不可用的。

在程序中,哪些对象可以作为GC Roots的对象呢?

1.虚拟机栈中引用的对象

2.方法区中类静态属性引用的对象

3.方法区中常量引用的对象

4.本地方法栈中引用的对象

上面两种判对象是否存活都需要用到引用,引用是一块代表另外一块内存起始地址的内存,在jdk 1.2后,Java对引用进行了扩充,增加了四种引用概念,分别是:

强引用:Object o = new Object() ; o就是一个强引用,这种引用的对象不会被回收的。

软引用:这类引用是还有用但是并非必须的对象,在JVM内存溢出前尝试回收该类引用关联的对象

弱引用:这类引用描述的是非必须对象的,它关联的对象在下一次回收内存时被回收,而不管内存是否足够

虚引用:这类引用也叫做幽灵引用或者幻影引用,它是最弱的一种引用关系。这类引用不会影响对象生存周期,也无法通过虚引用取到对象实例。

通过上面分析,我们可以知道在这些情况,需要进行内存回收

第三个问题,怎么回收内存呢?

这里就涉及到JVM的回收算法了,现在JVM主要有以下几种回收算法:

  1. 标记-清除算法

这种算法如它的名字一样,算法分为"标记"和"清除"两个过程:

a. 标记出所有需要回收的对象. 需要被回收的算法就是通过第二个问题里的方法实现的。

b.在标记完成后统一回收被标记的对象

这种算法有两个不足:

a. 效率不高,标记和清除都需要遍历整块内存的地址

b.清除标记的内存后回有大量的碎片产生,连续内存就不纯粹了

2.复制算法

这种算法有两种实现方式

第一种:将内存区域划分为两块相等的区域,每次只使用其中一块区域,当这块内存用完的时候,就将还存活的对象复制到另外一块上,然后把已经使用过的第一个块内存一次性清理掉。

这种方式只对一半的内存进行回收,也不用清理碎片,但是内存使用率下降了,只使用到了一半。

第二种:将内存分为三块区域,一块是较大的Eden空间和两块较小的Survivor空间,每次使用Eden和其中一块Survivor.当回收时,将Eden空间和Survivor空间的活着的对象一次性地复制到另外一块Survivor空间上,最后清理掉Eden和Survivor和用过的Survivor空间。如果回收时Survivor不够时将依赖其他内存区域(老年代)进行分配担保即将无法放入Survivor的对象加入老年代。Eden:Survivor = 8 :1

这种方式将内存利用率达到了90%,但是如果对象存活率比较高的时候就要进行较多的复制操作,降低了回收效率,所以这种算法多被用于回收年轻代。

3.标记-整理算法

标记整理算法是为了解决老年代对象存活几率大的问题而提出的一种i算法,和标记-清除算法类似也有标记过程,标记之后不是直接清除标记的内存,而是将存活的对象都往一个方向移动,然后直接清理掉端边界以为的内存。

这种回收方式对对象存活比较久的对象比较多的区域有很大的优势,移动之后就可以将内存空间换回来,不会浪费空间

4.分代回收算法

这种算法没有用到新的回收思想,只是将内存区域根据对象存活周期的不同划分为几块,一般有新生代,老年代,永久代(jdk8之后被移除),

然后根据各个年代的特点选择不同的已经有的回收算法。

年轻代:里面存放的对象生命周期都比较短,较多使用复制算法

老年代:里面存放的对象生命周期较长,一般选择标记-清除算法或者标记-整理算法来回收

相关文章
|
8天前
|
缓存 算法 Java
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
这篇文章详细介绍了Java虚拟机(JVM)中的垃圾回收机制,包括垃圾的定义、垃圾回收算法、堆内存的逻辑分区、对象的内存分配和回收过程,以及不同垃圾回收器的工作原理和参数设置。
28 4
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
|
7天前
|
存储 监控 算法
JVM调优深度剖析:内存模型、垃圾收集、工具与实战
【10月更文挑战第9天】在Java开发领域,Java虚拟机(JVM)的性能调优是构建高性能、高并发系统不可或缺的一部分。作为一名资深架构师,深入理解JVM的内存模型、垃圾收集机制、调优工具及其实现原理,对于提升系统的整体性能和稳定性至关重要。本文将深入探讨这些内容,并提供针对单机几十万并发系统的JVM调优策略和Java代码示例。
32 2
|
8天前
|
存储 Java
JVM知识体系学习四:排序规范(happens-before原则)、对象创建过程、对象的内存中存储布局、对象的大小、对象头内容、对象如何定位、对象如何分配
这篇文章详细地介绍了Java对象的创建过程、内存布局、对象头的MarkWord、对象的定位方式以及对象的分配策略,并深入探讨了happens-before原则以确保多线程环境下的正确同步。
23 0
JVM知识体系学习四:排序规范(happens-before原则)、对象创建过程、对象的内存中存储布局、对象的大小、对象头内容、对象如何定位、对象如何分配
|
11天前
|
Java API 对象存储
JVM进阶调优系列(2)字节面试:JVM内存区域怎么划分,分别有什么用?
本文详细解析了JVM类加载过程的关键步骤,包括加载验证、准备、解析和初始化等阶段,并介绍了元数据区、程序计数器、虚拟机栈、堆内存及本地方法栈的作用。通过本文,读者可以深入了解JVM的工作原理,理解类加载器的类型及其机制,并掌握类加载过程中各阶段的具体操作。
|
9天前
|
算法 Java
JVM进阶调优系列(4)年轻代和老年代采用什么GC算法回收?
本文详细介绍了JVM中的GC算法,包括年轻代的复制算法和老年代的标记-整理算法。复制算法适用于年轻代,因其高效且能避免内存碎片;标记-整理算法则用于老年代,虽然效率较低,但能有效解决内存碎片问题。文章还解释了这两种算法的具体过程及其优缺点,并简要提及了其他GC算法。
 JVM进阶调优系列(4)年轻代和老年代采用什么GC算法回收?
|
15天前
|
存储 算法 Java
【JVM】垃圾释放方式:标记-清除、复制算法、标记-整理、分代回收
【JVM】垃圾释放方式:标记-清除、复制算法、标记-整理、分代回收
39 2
|
15天前
|
存储 Java Linux
【JVM】JVM执行流程和内存区域划分
【JVM】JVM执行流程和内存区域划分
35 1
|
17天前
|
缓存 算法 Java
GC垃圾收集算法
这篇文章详细讨论了垃圾收集(GC)的几种算法,包括引用计数、可达性分析、标记-清除、标记-复制和标记-整理算法,并介绍了这些算法的优缺点和适用场景。
16 0
GC垃圾收集算法
|
5天前
|
存储 Kubernetes 架构师
阿里面试:JVM 锁内存 是怎么变化的? JVM 锁的膨胀过程 ?
尼恩,一位经验丰富的40岁老架构师,通过其读者交流群分享了一系列关于JVM锁的深度解析,包括偏向锁、轻量级锁、自旋锁和重量级锁的概念、内存结构变化及锁膨胀流程。这些内容不仅帮助群内的小伙伴们顺利通过了多家一线互联网企业的面试,还整理成了《尼恩Java面试宝典》等技术资料,助力更多开发者提升技术水平,实现职业逆袭。尼恩强调,掌握这些核心知识点不仅能提高面试成功率,还能在实际工作中更好地应对高并发场景下的性能优化问题。
|
10天前
|
算法 Java
JVM进阶调优系列(3)堆内存的对象什么时候被回收?
堆对象的生命周期是咋样的?什么时候被回收,回收前又如何流转?具体又是被如何回收?今天重点讲对象GC,看完这篇就全都明白了。