牛X的java程序员必备的GC基础知识, 面试肯定用的到

1. GC回收哪些内存区域呢?
  • 堆内存
    • 对象
    • 数组
  • 方法区
    • 该类所有的额实例都已经被回收, 也就是java堆中不存在该类的任何实例
    • 加载该类的ClassLoader已经被回收
    • 该类对应的java.lang.Class对象在任何地方没有被引用, 也无法通过反射访问该类方法。
    • 垃圾回收性价比比较低, 一般不回收
    • 废弃常量
    • 无用的类(需要同时满足以下三个条件)

2. GC是如何判断垃圾对象的?

1. 引用计数法

给对象的对象头中添加一个counter引用计数器,当该对象被引用时,counter+1,当不被引用时, counter-1

当对象的counter为0时,则说明该对象不可用,也就是垃圾对象。

缺点:容易引起对象之间的互相循环引用,造成死锁状态。

2. 可发性分析算法

通过GC Roots的对象为起点,向下搜索,能到达的对象为不可回收对象,不能到达的对象为需要回收的对象。Java中就是通过可达性分析算法来判定对象是否存活的。

牛X的java程序员必备的GC基础知识, 面试肯定用的到

3. 对象引用

无论是通过引用计数算法判断对象的引用数量,还是通过根搜索算法判断对象的引用链是否可达,判定对象是否存活都与“引用”有关。

在JDK1.2之前,Java中的引用的定义很传统:如果reference类型的数据中存储的数值代表的是另外一块内存的起始地址,就称这块内存代表着一个引用。这种定义很纯粹,但是太过狭隘,一个对象在这种定义下只有被引用或者没有被引用两种状态。我们希望能描述这样一类对象:当内存空间还足够时,则能保留在内存之中;如果

在JDK1.2之后,Java对引用的概念进行了扩充,将引用分为强引用(Strong Reference)、软引用(Soft Reference)、弱引用(Weak Reference)、虚引用(Phantom Reference),这四种引用强度依次逐渐减弱。

  • 强引用就是指在程序代码之中普遍存在,类似“Object obj = new Object()”这类的引用,只要强引用还存在,垃圾收集器永远不会回收掉被引用的对象。
  • 软引用用来描述一些还有用,但并非必需的对象。对于软引用关联着的对象,在系统将要发生内存溢出异常之前,将会把这些对象列进回收范围之中并进行第二次回收。如果这次回收还是没有足够的内存,才会抛出内存溢出异常。在JDK1.2之后提供了SoftReference类来实现软引用。
  • 弱引用也是用来描述非必需对象的,但是它的强度比软引用更弱一些,被弱引用关联的对象只能生存到下一次垃圾收集发生之前。当垃圾收集器工作时,无论当前内存是否足够,都会回收掉只被弱引用关联的的对象。在JDK1.2之后提供了WeakReference类来实现弱引用。
  • 虚引用(幽灵引用、幻影引用),他是最弱的一种引用关系。一个对象是否有虚引用的存在,完全不会对其生存时间构成影响,也无法通过虚引用来取得一个对象实例。为一个对象*设置虚引用关联的唯一目的就是希望能*在这个对象被收集器回收时收到一个系统通知。在JDK1.2之后,提供了PhantomReference类来实现虚引用。

4. 回收过程

第一次标记:使用可达性分析算法分析之后,判断对象不可达。

第二次标记:finalize()方法(上诉或者对象自我救赎的唯一方式)。该方法会被垃圾回收器去调 用,并且只会被调用一次。所以可以在finalize方法中,重新建立可达性关联,那么就完成了自我 救赎。否则被第二次标记。

3. 垃圾对象时怎么被回收的呢?

1. 复制回收算法

使用复制算法的垃圾回收步骤:

当Eden区发生垃圾回收之后,会将Eden区和Survivor其中的一块区域中的对象,复制到另一块 Survivor区域

然后将将Eden区和Survivor其中的一块区域中的对象完全清理掉。

缺点:内存分配时会浪费新生代的10%的空间。

2. 标记清除算法

最基本的算法,主要分为标记和清除2个阶段。首先标记出所有需要回收的对象,在标记完成后统一回收掉所有被标记的对象

缺点:

效率不高。
产生空间碎片。会产生大量不连续的内存碎片,会导致大对象可能无法分配,提前触发GC

3. 标记整理算法

老年代没有人担保,不能用复制回收算法。可以用标记-整理算法,标记过程仍然与“标记-清除”算法一样,然后让所有存活的对象都向一端移动,然后直接清理掉端边界以外的内存

4. 分代回收算法

当前商业虚拟机都是采用这种算法。根据对象的存活周期的不同将内存划分为几块。

新生代,每次垃圾回收都有大量对象失去,选择复制算法。
老年代,对象存活率高,无人进行分配担保,就必须采用标记清除或者标记整理算法

5. 内存分配担保

在JVM的内存分配时,也有这样的内存分配担保机制。就是当在新生代无法分配内存的时候,把新生代的对象转移到老生代,然后把新对象放入腾空的新生代。

4. GC方式有哪些?

minorGC

新生代的垃圾回收,很快就回收了,新生代回收的频率高

MajorGC

老年代的垃圾回收 比minorGC慢10倍

fullGC

整个JVM的垃圾回收。 整个堆(minorGC和majorGC)和方法区的垃圾回收。

  • system.gc()

  • 老年代不够用

  • 方法区不够用

当新生代的对象无法被老年代担保成功时

牛X的java程序员必备的GC基础知识, 面试肯定用的到

牛X的java程序员必备的GC基础知识, 面试肯定用的到

相关推荐