JVM垃圾回收(GC)

判断哪些对象是垃圾

  • 引用计数器算法(最早使用的算法)

    当我们在内存中创建一个对象时,会为他添加一个引用计数器,同时将引用计数器加1,当有新的引用器引用到子对象时,引用计数器再加1,当其中的一个引用销毁时,引用计数器减1,当引用计数器减为0 时,标志着此对象为垃圾对象,可被回收。

    存在问题:

    如图,当对象A引用对象B时,对象B的计数器加1,对象B再引用对象A,对象A的计数器也加1,但是并没有其他路径指向他们,此时的对象A和对象B已经成为垃圾对象,但由于二者的计数器都不为0,引用计数算法无法回收他们的内存。为了解决这个问题,可达性算法诞生了。

  • 可达性算法(根搜索算法)

    通过一系列的称为“GC Roots”的对象作为起始点,从这些节点开始向下搜索,搜索所走过的路径称为引用链(Reference Chain),当一个对象到GC Roots没有任何引用链相连(用图论的话来说,就是从GC Roots到这个对象不可达)时,则证明此对象是不可用的,即为垃圾对象

    此处的obj4、obj6、obj7、obj8都为不可达对象,即垃圾对象

引用的类型

  • 强引用

    Object obj = new Object();
    
  • 弱引用(不会阻碍内存的回收)

    需要被 WeakReference 类包装

    WeakReference<Object> wf = new WeakReference<Object>(obj);
    
  • 软引用

    需要被 SoftReference 类 包装

  • 虚引用

    需要被 PhantomReference 类 包装

最常用到的是强引用和弱引用

垃圾回收算法

  • 标记-清除算法

如图,从根节点引用A,A引用C,则B为不可达点,进行扫描,进入标记阶段,标记B,此时B为垃圾对象,接下来进入清除阶段,扫描整个空间并清除未标记的对象

缺点就是容易产生内存碎片,碎片太多可能会导致后续过程中需要为大对象分配空间时无法找到足够的空间而提前触发新的一次垃圾收集动作。

  • 复制算法

如图,从根节点引用A,将A放置在另一块空闲的内存中,B为不可达点,跳过,A引用C,将A放置在另一块空闲的内存中,当遍历完成后,将原来的内存清空,只保留了新的内存。

优点:对象少时,效率极高

缺点:需要一块内存

  • 标记-整理算法

如图,进行全局扫描,标记出可回收的对象,进入清除阶段,将可回收对象清除,将所有的存活对象向左端空闲处移动并更新对应空闲指针

触发回收

  1. Java虚拟机无法再为新的对象分配内存空间
  2. 手动调用System.gc()方法(强烈不推荐)
  3. 低优先级的GC线程,被运行时就会执行GC
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 1.什么是垃圾回收? 垃圾回收(Garbage Collection)是Java虚拟机(JVM)垃圾回收器提供...
    简欲明心阅读 90,088评论 17 311
  • GC中的垃圾特指于内存中不会再使用的对象,垃圾回收有很多算法:引用计数法标记压缩法复制算法分代,分区的思想 引用计...
    MicoCube阅读 2,823评论 0 1
  • 作者:一字马胡 转载标志 【2017-11-12】 更新日志 日期更新内容备注 2017-11-12新建文章初版 ...
    beneke阅读 6,616评论 0 7
  • 一. 垃圾回收的意义 在C++中,对象所占的内存在程序结束运行之前一直被占用,在明确释放之前不能分配给其它对...
    Stan_Z阅读 5,940评论 0 25
  • 原文阅读 前言 这段时间懈怠了,罪过! 最近看到有同事也开始用上了微信公众号写博客了,挺好的~给他们点赞,这博客我...
    码农戏码阅读 11,207评论 2 31