您的位置:

Java 垃圾回收器详解

一、垃圾回收器概述

Java 的垃圾回收机制可以自动回收程序中不用的对象,使开发者从手动释放内存的繁琐任务中解脱出来。Java垃圾收集器运行于Java虚拟机中,负责回收内存中的无用对象。

Java垃圾回收器设计的重要目标是减少内存泄漏问题,提高内存使用效率和程序执行效率。但是各个垃圾回收器的实现方式都不同,因此也会有各自的优缺点。

二、内存管理

Java堆是Java虚拟机中内存管理的主要组成部分。Java堆内存中存储着各个线程所需的对象。

public class Demo {
    private Object obj = new Object();
    public void method() {
        Object newObj = new Object();
    }
}

在JVM启动后,会先根据-Xms参数配置的大小来初始化一个Java堆,这个堆被等分为了新生代和老年代两个部分。新生代又被进一步分为了Eden区和两个Survivor区。大部分的对象都是在新生代中被分配,当新生代内存不足时,会触发Minor GC;而老年代主要存储程序中生存较久的对象,当老年代内存不足时,会触发Full GC。

三、垃圾回收算法

目前Java中常见的垃圾回收算法有标记-清除算法(Mark-Sweep)、复制算法(Copy)、标记-整理算法(Mark-Compact),以及最近常用的分代回收算法(Generational GC)。

如下是一个简单的Mark-Sweep算法的代码实现:

public class MarkSweep {
    void markAndSweep() {
        Set liveObjects = new HashSet<>();
        // 标记所有存活的对象
        mark(liveObjects);
        // 清除所有未标记的对象
        sweep(liveObjects);
    }
    void mark(Set liveObjects) {
        // 标记存活对象
    }
    void sweep(Set liveObjects) {
        // 清除未标记的对象
    }
}


四、常用垃圾回收器

1. Serial收集器

Serial收集器是最古老,也是最基础的收集器。它使用了单线程进行垃圾回收,属于串行收集。它适用于小型应用程序,对于单CPU的机器也非常适合使用,能够保证稳定性和高效率。

使用以下命令启用Serial收集器:

java -XX:+UseSerialGC Demo

2. Parallel收集器

Parallel收集器是Serial收集器的多线程版本,通过多线程的方式减少了垃圾回收的时间。它适用于多核服务器和多CPU的机器。但是它并不能保证高并发下的吞吐量和延迟。

使用以下命令启用Parallel收集器:

java -XX:+UseParallelGC Demo

3. CMS收集器

CMS(Concurrent Mark Sweep)收集器是一种具有低停顿,高并发的垃圾收集器。它采用了标记-清除算法,可以有效地减少垃圾收集器的暂停时间。但是还有一些问题,比如可能会因碎片问题而导致长时间的Full GC暂停等。

使用以下命令启用CMS收集器:

java -XX:+UseConcMarkSweepGC Demo

4. G1收集器

G1(Garbage First)收集器是Java 7引入的全新垃圾收集器。G1通过划分内存块,进行垃圾回收,可控制垃圾回收时间和内存占用,它采用了分代回收的思想,将Java堆划分为不同的区域,每个区域独立进行垃圾收集。相比于CMS收集器,G1收集器能够更加有效地避免碎片问题。

使用以下命令启用G1收集器:

java -XX:+UseG1GC Demo

五、常用命令参数

以下是一些常用的命令参数:

  • -XX:+PrintGCDetails:在GC结束后输出GC的详细信息。
  • -XX:+PrintGCDateStamps:打印GC发生时间。
  • -XX:+UseSerialGC:使用Serial收集器。
  • -XX:+UseParallelGC:使用Parallel收集器。
  • -XX:+UseConcMarkSweepGC:使用CMS收集器。
  • -XX:+UseG1GC:使用G1收集器。
  • -Xms:Java虚拟机最小堆内存大小。
  • -Xmx:Java虚拟机最大堆内存大小。
顶部