JVM:垃圾收集器与对象的"存活"问题
Posted magic-sea
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了JVM:垃圾收集器与对象的"存活"问题相关的知识,希望对你有一定的参考价值。
垃圾收集器
垃圾收集(Garbage Collection,GC)。当需要排查各种内存溢出、内存泄露问题时,当垃圾收集成为系统更高并发量的瓶颈时,我们需要去了解GC和内存分配。
检查对象的"存活"状态
- 引用计数法:每当有一个地方引用对象时,计数器加1;当引用值失效,减1;任何时刻计数器为0,则表示对象不可能再被使用。缺点就是很难解决对象之间相互循环引用的问题。(如a.instance=b;b.instance=a)
- 可达性分析算法:这个算法的思想是通过一系列成为“GC Roots”的对象作为起始点,从这些结点向下搜索,搜索走过的路径成为引用链,当一个对象到GC Roots没有任何引用链相连时,则证明该对象不可用。在Java语言中,可作为GC Roots的对象包括下面几种:
- 虚拟机栈(栈帧中的本地变量表)中引用的对象;
- 方法区中静态属性引用的对象;
- 方法区中常量引用的对象;
- 本地方法中JNI(即一般说法中的Native方法)引用的对象。
- 再谈引用:之前两个方法中对象的存活时都是与“引用”相关。JDK1.2后,Java对引用概念进行了扩充,将引用分为强引用、软引用、弱引用、虚引用这四种,四种强度依次逐渐减弱。
- 强引用是代码中普遍存在的,类似于Object o=new Object()这类引用,只要强引用还在,垃圾收集器就永远不会回收被引用的对象。
- 软引用是描述一些还有用但非必须的对象,对于软引用关联的对象,在系统将要发生内存溢出异常之前,会把这些对象列入回收范围之中,并进行第二次回收。若是不行才会报内存溢出异常。
- 弱引用是描述非必需对象的,它的强度比软引用更弱一些,被弱引用的对象只能生存到下一个垃圾收集之前(无论内存是否充足,都会被回收)。
- 虚引用也被称为幽灵引用或者幻影引用,是最弱的一种引用关系。一个对象是否有虚引用的存在,完全不会对其生存时间构成影响,也无法对虚引用来获取一个对象的实例。为对象设置虚引用关联的唯一目的是能在这个对象能被收集器回收时收到一个系统通知。
- 生存还是死亡:在可达性分析中不可达对象也不是"非死不可",这时候它们处于缓刑阶段,要宣告一个对象的死亡,至少要经历两次标记过程。如果对象在进行可达性分析后没有与GC Roots相连的引用链,那它将会被第一次标记并且进行一次筛选,筛选条件是此对象中是否有必要执行finalize()方法。当对象没有覆盖finalize()方法,或finalize()方法已经被虚拟机调用过,虚拟机江浙两种情况都视为"没有必要执行"。如果这个对象被判定有必要执行finalize()方法,那这个对象会被放置在F-Queue的队列之中,稍后由虚拟机自动建立的、低优先级的Finalizer线程去执行(触发这个方法,但不能保证运行完,因为可能会有死循环或者执行速度缓慢)它。finalize()方法是对象摆脱被回收的最后一次机会(只要重新链接上引用链),这时若是没能拯救自己,就会被第二次标记。
- 回收方法区:Java虚拟机规范没有要求在方法区进行垃圾收集,而且在方法区回收垃圾效率比较低。永久代的垃圾收集主要分为2部分,废弃常量和无用的类。JDK8移除了了永久代,相对的元空间metaspace会不会有gc来收集垃圾呢?GC默认会在元空间变满时对其进行回收,它还会动态增加(如果允许)分配给元数据内存。由于元空间的动态扩展(这是permgen无法做到的),启用CMSClassUnloadingEnabled,会使得GC在元空间的效率比永久代更好。
以上是关于JVM:垃圾收集器与对象的"存活"问题的主要内容,如果未能解决你的问题,请参考以下文章