美文网首页
Java GC 必知必会

Java GC 必知必会

作者: 我不想成为胖子 | 来源:发表于2021-03-10 08:42 被阅读0次

1. Java 如何标识垃圾

常用的标识算法主要是两类,一是计数器引用法,二是可达性分析(根搜索算法)。

  • 计数器引用法

  • 可达性分析
    基本思路:

    • 已根对象集合为起点,按照从上之下的方式搜索被根对象集合所连接的目标对象是否可达。
    • 使用可达性分析算法后,内存中的存活对象都会被根对象集合直接或间接连接着,搜索所走过的路径成为引用链.
image.png

2. GC ROOTS 包含哪些?

  • 虚拟机栈中引用的对象
    》 比如各个线程中被调用的方法中使用的参数,局部变量
  • 本地方法栈引用的对象
  • 方法区中类静态属性引用的对象
    》 java类的引用类型静态变量
  • 方法区中常量引用的对象
    》 字符串常量池里面的应用
  • 被synchronized持有的对象
  • java虚拟机内部的引用
  • 反映java虚拟机内部情况的JMXBean,JVMTI中的回调,本地代码缓存等。

特殊情况:
有对象“临时性”的加入,共同构成完成GC Roots集合。比如分代收集和局部回收(Partial GC)

image.png image.png

3. 为什么会产生STW

如果要使用可达性分析算法来判断内存是否可回收,那么分析工作必须在一个能保障一致性的快照中进行。这点不满足的话,那分析的结果准确性也就无法保障。

所以就会产生了Stop The World的一个重要原因。

4. 对象的finalization机制

  • 用于开发人员对对象被销毁前的自定义逻辑处理。通常用于对象被回收时,进行资源释放和清理的工作。

  • 关闭文件流,关闭数据库连接等。

finalization为什么不要手工调用

  • 有可能导致对象复活
  • finalization的执行时间点不确定。在极端情况下,如果不发生GC,那么finalization将永远不会被执行。

finalization 导致对象复活的情况
在虚拟机定义中,对象存在三种状态
1. 可触及的:从GC ROOT可达
2. 可复活的:对象多有的引用都被释放了,但是对象可能在finalization() 被复活。
3. 不可触及的:对象的finalization()被调用,并且没有复活。那么就会就如不可触及的状态。不可触及的对象就一定会被回收。因为finalization只会被调用一次。

5. 如何判断一个对象是否可以被回收?

判断一个对象能否被收回,至少要经历两次判断。

  1. 如果objA到GC Roots没有引用链,则进行第一次标记
  2. 判断对象是否有必要执行finalization()方法
    • 如果没有重写finalization()或者finalization()已经被执行过了,则被判断为不可触及的。
    • 如果对象重写了finalization()方法,且还未被执行过。那么objA会被插入到F-Queue队列中,由一个虚拟机自动创建的低优先级的finalizer线程触发其finalization()。
  • finalization()是对象逃脱死亡的最后机会。稍后GC会对F-Queue 队列中的对象进行二次标记。 如果objA在此时与引用链上的任何一个对象建立了联系,那么objA会被移出“即将回收”集合。

6. JVM GC 清除阶段的算法

6.1 标记-清除算法(Mark-Sweep)

image.png image.png
  • 优点: 比较容易理解
  • 缺点
  1. 效率不算高
  2. 会产生内存碎片,还需要额外的空间维护一个空闲列表。

6.2 复制算法(Copying)

image.png image.png

优点:

  • 保证空间连续,不会出现碎片问题
  • 实现相对简单,运行高效

缺点:

  • 需要两倍的空间
  • 当系统存活的对象数量很多时,性能较低。(所以只能用于新生代)

6.3 标记-压缩算法(Mark-Compact)

image.png image.png

优点

  • 没有内存碎片
  • 减少了内存的浪费

缺点

  • 从效率来说,标记整理算法要低于复制算法
  • 移动对象的同时,如果对象被其他对象引用,则还需调整引用地址。
  • STW的时间相比于其他要长一些,因为涉及到对象的移动和引用更新。
image.png

7. GC 分带收集算法&增量收集算法&分区算法

分带收集算法

image.png

增量收集算法

image.png

分区算法

image.png

8. System.gc()和Runtime.getRuntime().gc()的理解

  • 默认情况下,通过System.gc()和Runtime.getRuntime().gc()显示调用时,会触发Full GC,同时对老年代和新生代进行回收,尝试释放对象占用的内存。
  • System.gc()无法保证对垃圾回收器的调用
  • Sysmte.gc()等同与Runtime.getRuntime().gc()

gc与slot的关系

image.png

在这个案例中 buffer所占用空间不会被回收,因为slot=1的位置还是被buffer这个变量所占用。在gc时,属于GC Root可达的情况。

9. 程序的并行与并发

image.png

垃圾回收器的并行与并发

image.png image.png

10. GC 中的安全点与安全区域的说明

  • Safepoint


    image.png

如何让线程在安全点中断:


image.png
  • SafeRegion


    image.png

实际执行流程

image.png

11. java引用:强,软,弱,虚

  • 强引用:不回收

    1. 强引用的对象的GC Root可达的,所以垃圾回收器永远不会回收。
    2. 强引用也是造成内存泄漏的主要原因之一。
  • SoftReference 软引用:内存不足时及回收

    1. 用于描述一下有用但是非必须的对象。只被弱引用关联者的对象,在系统将要发生内存溢出前,会把这些对象列入回收范围内进行二次回收。如果这次回收还是没有足够的内存,才会报出OOM。
    2. 一般使用场景是:如mybaits中的本地缓存
  • WeakReference 弱引用:发现即回收

  1. 只被弱引用关联的对象,只能存活到下一次GC发生为止。
  2. 常用的实现类是WeekHashMap,在Tomcat中作为了一个LRU的cache实现。

软引用和弱引用的区别点:


image.png
  • PhantomReference 虚引用:
    image.png
  1. 使用场景:追踪垃圾回收
  • ** 终结器引用:**
    对象finalization的底层实现


    image.png

相关文章

  • Java GC 必知必会

    1. Java 如何标识垃圾 常用的标识算法主要是两类,一是计数器引用法,二是可达性分析(根搜索算法)。 计数器引...

  • Java GC 必知必会-垃圾回收器篇

    1. 垃圾回收器的分类 按线程数:串行垃圾和并行 按工作模式:并发式和独占式并发式:垃圾回收线程与用户线程交替工作...

  • 优秀参考文章--性能优化

    Android 性能优化必知必会Android 性能优化必知必会

  • Java面试必知必会

    试题来源 简历篇 请自我介绍 请介绍项目 基础篇 基本功 面向对象的特征 final, finally, fina...

  • 阿里P8 架构师 浅谈 HashMap 的机制与原理

    前言 HashMap可以说是Java程序员必知必会的东西了,基础中的基础,必会中的必会,重点中的重点;所谓万丈高楼...

  • Java/Android Exchanger 必知必会

    问:你知道 Java 的 Exchanger 吗?简单说说其特点及应用场景? 答:Exchanger 是 JDK ...

  • Java - 并发编程必知必会

    一、概述 在操作系统的学习中我们知道,随着不断提出的新的应用需求,计算机体系结构的不断发展,操作系统也在不断地发展...

  • Java线程池必知必会

    Java线程池必知必会 知其然更要知其所以然。同学们在使用线程池的过程中,有没有理解我们为什么要这么做,怎么配置线...

  • 必知必会

    1、斜杠(/)和反斜杠(\) 文件路径 统一使用 / (斜杠)与网址路径保持一致。在有些特殊的情况下,灵活变通。事...

  • Java多线程与并发编程在面试中的超高频题!

    Java多线程与并发编程高频问题 《[Java面试必知必会]》系列已经更新两章了,Java基础知识面试高频考点和J...

网友评论

      本文标题:Java GC 必知必会

      本文链接:https://www.haomeiwen.com/subject/xnobqltx.html