JVM垃圾回收入门看这一篇文章就够了

JVM 内存分配与回收

java的内存的自动管理的,Java 自动内存管理最核心的功能是 堆 内存中对象的分配与回收。
Java 堆也称gc堆,现在收集器基本都采用分代垃圾收集算法,所以 Java 堆还可以细分为:新生代和老年代:再细致一点有:Eden 空间、From Survivor、To Survivor 空间等等.
JVM垃圾回收入门看这一篇文章就够了
对象都会首先在 Eden 区域分配,在一次新生代垃圾回收后,如果对象还存活,则会进入 s0,Eden区被清空, 等Eden区再满了,再次触发一次Minor GC,Eden和S0中的存活对象又会被复制送入第二块survivor s1,S0和Eden被清空,然后下一轮S0与S1交换角色,如此循环往复。如果对象的复制次数达到16次,该对象就会被送到老年代中.对象晋升到老年代的年龄阈值,可以通过参数 -XX:MaxTenuringThreshold 来设置。
JVM垃圾回收入门看这一篇文章就够了
以上是堆的内存分配常见的三种策略.在加上一条,如果survivor区域的相同年龄对象大于survivor的一半,将会直接分配到老年代中.

-XX: PrintGCDetails 打印堆内存回收的日志.

新生代 GC(Minor GC):指发生新生代的的垃圾收集动作,Minor GC 非常频繁,回收速度一般也比较快。
老年代 GC(Major GC/Full GC):指发生在老年代的 GC,出现了 Major GC 经常会伴随至少一次的 Minor GC(并非绝对),Major GC 的速度一般会比 Minor GC 的慢 10 倍以上。

如何判断对象已经死亡

  • 引用计数法 给对象中添加一个引用计数器,每当有一个地方引用它,计数器就加 1;当引用失效,计数器就减 1;任何时候计数器为 0 的对象就是不可能再被使用的。但是目前主流的虚拟机中并没有选择这个算法来管理内存,其最主要的原因是它很难解决对象之间相互循环引用的问题。
  • 可达性分析算法 GC Roots 的对象作为起点,从这些节点开始向下搜索,节点所走过的路径称为引用链,当一个对象到 GC Roots 没有任何引用链相连的话,则证明此对象是不可用的。

上述两种方法都与引用有关:

  1. 强引用 : 必不可少的生活用品,垃圾回收器绝不会回收它。
  2. 软引用 : 可有可无的生活用品。如果内存空间足够,垃圾回收器就不会回收它,如果内存空间不足了,就会回收这些对象的内存。
  3. 弱引用 : 不管当前内存空间足够与否,都会回收它的内存。
  4. 虚引用 : 在任何时候都可能被垃圾回收

注意: 不可达对象并不是非死不可,要真正宣告一个对象死亡,至少要经历两次标记过程,第一次标记并且进行一次筛选,筛选的条件是此对象是否有必要执行 finalize 方法。当对象没有覆盖 finalize 方法,或 finalize 方法已经被虚拟机调用过时,虚拟机将这两种情况视为没有必要执行。被判定为需要执行的对象将会被放在一个队列中进行第二次标记,除非这个对象与引用链上的任何一个对象建立关联,否则就会被真的回收。

  • 如何判断一个常量是废弃常量 1.实例都已经被回收 2.ClassLoader 已经被回收 3.java.lang.Class 对象没有在任何地方被引用

垃圾收集算法

JVM垃圾回收入门看这一篇文章就够了

  • 标记-清除算法 先标记在清除,问题:效率低,占用空间(标记清除后会产生大量不连续的碎片)
  • 复制算法 为了解决标记-清除算法的效率问题,它可以将内存分为大小相同的两块,每次使用其中的一块。当这一块的内存使用完后,就将还存活的对象复制到另一块去,然后再把使用的空间一次清理掉。
  • 标记-整理算法
  • 分代收集算法 : 在新生代中,每次收集都会有大量对象死去,所以可以选择复制算法,而老年代的对象存活几率是比较高的,而且没有额外的空间对它进行分配担保,所以我们必须选择“标记-清除”或“标记-整理”算法进行垃圾收集。

垃圾收集器

JVM垃圾回收入门看这一篇文章就够了
Serial (串行)收集器 : 单线程收集器,在进行垃圾收集工作的时候必须暂停其他所有的工作线程.Serial 收集器由于没有线程交互的开销,可以获得很高的单线程收集效率。
ParNew 收集器 : Serial 收集器的多线程版本,能与 CMS 收集器配合工作,在 Server 模式下的虚拟机的首要选择

并行和并发概念补充:
并行(Parallel) :指多条垃圾收集线程并行工作,但此时用户线程仍然处于等待状态。
并发(Concurrent):指用户线程与垃圾收集线程同时执行(但不一定是并行,可能会交替执行),用户程序在继续运行,而垃圾收集器运行在另一个 CPU 上。

Parallel Scavenge 收集器 : 几乎和ParNew都一样,吞吐量高.
Serial Old 收集器 : Serial 收集器的老年代版本
Parallel Old 收集器 : Parallel Scavenge 收集器的老年代版本
CMS 收集器 : 非常符合在注重用户体验的应用上使用,让垃圾收集线程与用户线程(基本上)同时工作。
G1 收集器 : 面向服务器的垃圾收集器,以极高概率满足 GC 停顿时间要求的同时,还具备高吞吐量性能特征.

谈谈 JVM 类加载过程是怎样的?

  1. 加载 获取二进制字节流
  2. 验证 文件验证(验证字节流是否符合class规范),元数据验证(字节码描述进行语义分析)
  3. 准备 为类设置内存[仅包括类变量(static)]和初始值
  4. 解析 将常量池内的符号引用替换为直接引用的过程
  5. 初始化 执行类构造器 ()方法的过程。