jvm垃圾回收算法

1 、什么是垃圾回收?

程序运行需要申请内存资源,对象资源使用完如果不及时处理就会一直占用内存资源,最后导致内存溢出,所以对内存资源的管理是非常重要的。

1.1、C/C++语言的垃圾回收

在C/C++语言中,没有自动垃圾回收机制,是通过new关键字申请内存资源,通过delete
关键字主动释放内存资源
如果程序员忘记写delete进行释放,那么申请的对象将一直占用内存资源,最终导致内存溢出。

1.2、Java语言的垃圾回收

在Java语言中,有自动的垃圾回收机制,也就是我们熟悉的GC。
程序员只需要关心内存的申请即可,系统自动释放内存资源。自动垃圾回收的算法就非常重要了,因为算法不合理而导致内存资源一直没有释放,同样也可能会导致内存溢出的。
当然,除了Java语言,C#、Python等语言也都有自动的垃圾回收机制。

2 、常见垃圾回收算法

常见的垃圾回收算法有:引用计数法、标记清除法、标记压缩法、复制算法、分代算法等

2.1、引用计数法

引用计数是历史最悠久的一种算法,假设有一个对象A,任何一个对象对A的引用,那么对象A的引用计数器+1,当引用失败时,对象A的引用计数就-1,如果对象A的计数器的值为0,就说明对象A没有引用了,可以被回收。
优点:
实时性较高,运行时根据对象的计数器是否为0,就可以直接回收
在垃圾回收过程中,应用无需挂起。
区域性,更新对象的计数器时,只是影响到该对象,不会扫描全部对象。
缺点:
每次对象被引用时,都需要去更新计数器,有一点时间开销。
浪费 CPU资源,即使内存够用,仍然在运行时进行计数器的统计。
无法解决循环引用问题。(最大的缺点

2.2、标记清除法

标记清除算法,是将垃圾回收分为2个阶段,分别是标记和清除。
标记:从根节点开始标记引用的对象。
清除:未被标记引用的对象就是垃圾对象,可以被清理。
图例说明1
在这里插入图片描述
图例说明2
在这里插入图片描述

假设这会儿有效内存空间耗尽了,JVM将会停止应用程序的运行(stop the world)并开启GC线程,然后开始进行标记工作,按照根搜索算法,标记完以后,所有从 root对象可达的对象就被标记为了存活的对象,此时已经完成了第一阶段标记。没有被标记的对象将会回收清除掉,而被标记的对象将会留下,并且会将标记位重新归0。接下来就不用说了,唤醒停止的程序线程,让程序继续运行即可。
优点:
解决了引用计数算法中的循环引用的问题
缺点:
效率较低(2次遍历所有对象),程序停顿。标记和清除两个动作都需要遍历所有的对象,并且在 GC时,需要停止应用程序,对于交互性要求比较高的应用而言这个体验是非常差的。
内存碎片化。通过标记清除算法清理出来的内存,碎片化较为严重,因为被回收的对象可能存在于
内存的各个角落,所以清理出来的内存是不连贯的。

2.3、标记压缩算法

标记压缩算法是在标记清除算法的基础之上,做了优化改进的算法。在清理阶段,并不是简单的清理未标记的对象,而是将存活的对象压缩到内存的一端,然后清理边界以外的垃圾,从而解决了碎片化的问题。
在这里插入图片描述
优点:
解决了标记清除算法的内存碎片化的问题
缺点:
标记压缩算法多了移动对象内存位置的步骤,其效率也有有一定的影响。

2.4、复制算法

复制算法的核心就是,将原有的内存空间一分为二,每次只用其中的一块,在垃圾回收时,将正在使用的对象复制到另一个内存空间中,然后将该内存空间清空,交换两个内存的角色,完成垃圾的回收。
如果内存中的垃圾对象较多,存活对象少,需要复制的对象就较少,这种情况下适合使用该方式并且效率比较高,反之,则不适合。
优点:
在垃圾对象多而存活对象少的情况下(年轻代),效率较高
清理后,内存无碎片
缺点:
在垃圾对象少而存活对象多的情况下(老年代),效率低,不适用
分配的2块内存空间,在同一个时刻,只能使用一半,内存使用率较低(使用率只有50%)

2.5、分代算法

根据回收对象的特点进行选择,在jvm中,年轻代适合使用复制算法,老年代适合使用标记清除或标记压缩算法。
JVM中年轻代内存空间:一般eden区:survivor区=8:2(from区,to区一样大)

在GC开始的时候,对象只会存在于Eden区和名为“From”的Survivor区,Survivor区“To”是空的。

紧接着进行GC,Eden区中所有存活的对象都会被复制到“To”,而在“From”区中,仍存活的对象会根据他们的年龄值来决定去向。年龄达一定值(年龄阈值最大15,因为只有4bit,可以通过-XX:MaxTenuringThreshold来设置)的对象会被移动到年老代中,没有达到阈值的对象会被复制到“To”区域。

经过这次GC后,Eden区和From区已经被清空。这个时候,“From”和“To”会交换他们的角色,也就是新的“To”就是上次GC前的“From”,新的“From”就是上次GC前的“To”。不管怎样,都会保证名为To的Survivor区域是空的

GC会一直重复这样的过程,直到“To”区被填满,“To”区被填满之后,会将所有对象移动到年老代中。

jvm堆内存模型

在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值