JVM虚拟机垃圾回收机制

简介: JVM虚拟机垃圾回收机制



垃圾回收机制

判断是否存活算法

java语言和我们之前学的c/c++不同,c/c++可以手动进行内存释放,那样随时随地就可以释放不必要的内存,减少内存溢出,但我们java的内存是有jvm虚拟机自行释放的,当一个对象不再被使用的时候,jvm会自行释放并回收内存,那么它是怎么样来判断一个对象是否存活的呢?想必这里一定有着一套耐人寻味的算法

引用计数法

1·每个对象都包含一个引用计数器,用于存放引用计数(其实就是存放被引用的次数)
2.每当有一个地方引用此对象时,引用计数+1
3·当引用失效(比如离开了局部变量的作用域或是引用被设定为null)时,引用计数-1
局部变量的作用域:从定义开始,到其所在的大括号结束为止。
4·当引用计数为0时,表示此对象不可能再被使用,因为这时我们已经没有任何方法可以得到此对象的引用了
但这种方法却存在一个问题,如果两个对象相互引用呢?

public class jvm {
    public static void main(String[] args) {
        Test a = new Test();
        Test b = new Test();
        a.another = b;
        b.another = a;
        a = b = null;
    }
    private static class Test{
        Test another;
    }
}

按照引用计数算法,那么当出现以上情况时,虽然我们无法在得到此对象的引用了,并且此对象我们也无需再使用,但是由于这两个对象直接存在相互引用的情况,那么引用计数器的值将会永远是1,但是实际上此对象已经没有任何用途了。所以引用计数法并不是最好的解决方案。

可达性分析法

首先每个对象的引用都有机会成为树的根节点(GC Roots),可以被选定作为根节点条件如下:
1.位于虚拟机栈的栈帧中的本地变量表中所引用到的对象(其实就是我们方法中的局部变量)同样也包括本地方法栈中JNI引用的对象。
2.类的静态成员变量引用的对象。
3.方法区中,常量池里面引用的对象,比如我们之前提到的String类型对象。

4.被添加了锁的对象(比如synchronized关键字)
5.虚拟机内部需要用到的对象。


一旦已经存在的根节点不满足存在的条件时,那么根节点与对象之间的连接将被断开。此时虽然对象1仍存在对其他对象的引用,但是由于其没有任何根节点引用,所以此对象即可被判定为不再使用。比如某个方法中的局部变量引用,在方法执行完成返回之后:



可以解释成当一个对象为null就是没有指向任何GC Roots则证明这个对象是不可以再使用的了


最终判定

我们在上面讲到的可达性分析法,他只能告诉我们jvm可以去回收,但真正的想回收还必须经过我们的最终判定,并不是说算法判定可以回收之后,这个对象一定会被回收
Object类里面有一个方法名为finalize(),这个方法就是最终判定方法,如果子类里面重写了此方法,放该子类对象被判定为可回收的时候,会进行二次确认,如果确认之后对象依然不满足被回收的条件的时候,那么该对象依然逃脱不了被回收的命运

示例代码:

public class jvm {
    private static Test a;
    public static void main(String[] args) throws InterruptedException {
        a = new Test();
        a = null;
        System.gc();
        Thread.sleep(1000);
        System.out.println(a);
    }
    private static class Test{
        @Override
        protected void finalize() throws Throwable {
            System.out.println(this+"开启了他的");
            a = this;
        }
    }
}

运行结果:


但切记,该方法只会起到一次效果,如果之后再出现这种情况就不会奏效了
示例代码:

public class jvm {
    private static Test a;
    public static void main(String[] args) throws InterruptedException {
        a = new Test();
        a = null;
        System.gc();
        Thread.sleep(1000);
        System.out.println(a);
        a = null;
        System.gc();
        Thread.sleep(1000);
        System.out.println(a);
    }
    private static class Test{
        @Override
        protected void finalize() throws Throwable {
            System.out.println(this+"开启了他的");
            a = this;
        }
    }
}

运行结果:


之所有代码中要加上Thread.sleep()是因为finalize()方法是新开了一个线程执行的,并且该线程的优先级比较低,以免主线程已经执行完了,该线程还在执行着
当然,除了堆中的对象以外,方法区中的数据也是可以被垃圾回收的,但是回收条件比较严格,这里就暂时不谈了


垃圾回收算法

前面我们介绍了对象存活判定算法,现在我们已经可以准确地知道堆中的哪些对象可以被回收了,那么,接下来就该考虑如何对对象进行回收了,垃圾收集器会不定期地检查堆中的对象,查看它们是否满足被回收的条件。我们该如何对这些对象进行回收,是一个一个判断是否需要回收吗?

分代收集机制

实际上,如冤我们对堆中的每一个对象都依次判断是否需要回收,这样的效率其实是很低的,那么有没有更好地回收机制呢?
第一步,我们可以对堆中的对象进行分代管理。
比如某些对象,在多次垃圾回收时,都未被判定为可回收对象,我们完全可以将这一部分对象放在一起,并让垃圾收集器减少回收此区域对象的频率,这样就能很好地提高垃圾回收的效率了。
因此,Java虚拟机将堆内存划分为新生代、老年代和永久代(其中永久代是HotSpot虚拟机特有的概念,在JDK8之前方法区实际上就是采用的永久代作为实现,而在JDK8之后,方法区由元空间实现,并且使用的是本地内存,容量大小取决于物理机实际大小
这里我们主要讨论的是新生代和老年代。
不同的分代内存回收机制也存在一些不同之处,在HotSpot虚拟机中,新生代被划分为三块,一块较大的Eden空间和两块较小的Survivor空间,默认比例为8:∶ 1∶1,老年代的GC评率相对较低,永久代一般存放类信息等〈其实就是方法区的实现)如图所示:


Minor GC - 次要垃圾回收,主要进行新生代区域的垃圾回收

-触发条件:新生代Eden区容量已满的时候

Major GC -主要垃圾回收,主要进行老年代的垃圾收集

Full DC -完全垃圾回收,对整个java堆内存和方法区进行垃圾回收

·触发条件1:每次晋升到老年代的对象平均大小大于老年代剩余空间

·触发条件2: Minor GC后存活的对象超过了老年代剩余空间
·触发条件3:永久代内存不足(JDK8之前)
·触发条件4∶手动调用System.gc(方法

空间分配担保

考虑一种极端情况(正常情况下新生代的回收率是很高的,所以不用太担心会经常出现这种情况)就是当新生代的Eden区经历过一次回收之后,仍然存在大量的对象。
这时就需要用到空间分配担保机制了,可以把Survivor区无法容纳的对象直接送到老年代,让老年代进行分配担保(当然老年代也得装得下才行)在现实生活中,贷款会指定担保人,就是当借款人还不起钱的时候由担保人来还钱。

好,那既然新生代装不下就丢给老年代,那么要是老年代也装不下新生代的数据呢?这时,老年代肯定担保人是当不成了,那么这样的话,首先会判断一下之前的每次垃圾回收进入老年代的平均大小是否小于当前老年代的剩余空间,如果小于,那么说明也许可以放得下(不过也仅仅是也许,依然有可能放不下,因为判断的实际上只是平均值,万一这一次突然非常大呢),否则,会先来一次FullGC,进行一次大规模垃圾回收,来尝试腾出空间,再次判断老年代是否有空间存放,要是还是装不下,直接抛出0OM错误,摆烂。


目录
相关文章
|
1月前
|
监控 算法 Java
Java虚拟机(JVM)垃圾回收机制深度剖析与优化策略####
本文作为一篇技术性文章,深入探讨了Java虚拟机(JVM)中垃圾回收的工作原理,详细分析了标记-清除、复制算法、标记-压缩及分代收集等主流垃圾回收算法的特点和适用场景。通过实际案例,展示了不同GC(Garbage Collector)算法在应用中的表现差异,并针对大型应用提出了一系列优化策略,包括选择合适的GC算法、调整堆内存大小、并行与并发GC调优等,旨在帮助开发者更好地理解和优化Java应用的性能。 ####
43 0
|
14天前
|
算法 网络协议 Java
【JVM】——GC垃圾回收机制(图解通俗易懂)
GC垃圾回收,标识出垃圾(计数机制、可达性分析)内存释放机制(标记清除、复制算法、标记整理、分代回收)
|
1月前
|
存储 监控 算法
深入探索Java虚拟机(JVM)的内存管理机制
本文旨在为读者提供对Java虚拟机(JVM)内存管理机制的深入理解。通过详细解析JVM的内存结构、垃圾回收算法以及性能优化策略,本文不仅揭示了Java程序高效运行背后的原理,还为开发者提供了优化应用程序性能的实用技巧。不同于常规摘要仅概述文章大意,本文摘要将简要介绍JVM内存管理的关键点,为读者提供一个清晰的学习路线图。
|
1月前
|
存储 监控 算法
Java虚拟机(JVM)垃圾回收机制深度解析与优化策略####
本文旨在深入探讨Java虚拟机(JVM)的垃圾回收机制,揭示其工作原理、常见算法及参数调优方法。通过剖析垃圾回收的生命周期、内存区域划分以及GC日志分析,为开发者提供一套实用的JVM垃圾回收优化指南,助力提升Java应用的性能与稳定性。 ####
|
2月前
|
机器学习/深度学习 监控 算法
Java虚拟机(JVM)的垃圾回收机制深度剖析####
本文深入探讨Java虚拟机(JVM)的垃圾回收机制,揭示其工作原理、常见算法、性能调优策略及未来趋势。通过实例解析,为开发者提供优化Java应用性能的思路与方法。 ####
57 1
|
存储 算法 Java
Java高级之虚拟机垃圾回收机制
博客出自:http://blog.csdn.net/liuxian13183,转载注明出处! All Rights Reserved ! 区别于C语言手动回收,Java自动执行垃圾回收,但为了执行高效,需要了解其策略,更好的去应用。
1240 0
|
2月前
|
Ubuntu 网络安全 虚拟化
VMware虚拟机ping不通原因排查及分析
下面以 VMware 虚拟机为例进行介绍。
1189 3
|
2月前
|
存储 SQL 数据库
虚拟化数据恢复—Vmware虚拟机误还原快照的数据恢复案例
虚拟化数据恢复环境: 一台虚拟机从物理机迁移到ESXI虚拟化平台,迁移完成后做了一个快照。虚拟机上运行了一个SQL Server数据库,记录了数年的数据。 ESXI虚拟化平台上有数十台虚拟机,EXSI虚拟化平台连接了一台EVA存储,所有的虚拟机都存放在EVA存储上。 虚拟化故障: 工组人员误操作将数年前迁移完成后做的快照还原了,也就意味着虚拟机状态还原到数年前,近几年数据都被删除了。 还原快照相当于删除数据,意味着部分存储空间会被释放。为了不让这部分释放的空间被重用,需要将连接到这台存储的所有虚拟机都关掉,需要将不能长时间宕机的虚拟机迁移到别的EXSI虚拟化平台上。
116 50
|
3月前
|
安全 虚拟化 数据中心
Xshell 连接 VMware虚拟机操作 截图和使用
Xshell 连接 VMware虚拟机操作 截图和使用
88 4
|
3月前
|
Linux 虚拟化
vmware虚拟机安装2024(超详细)
vmware虚拟机安装2024(超详细)
437 6