Java 8作为Java语言的重要里程碑,不仅引入了Lambda表达式、Stream API等新特性,还在垃圾回收(Garbage Collection, GC)方面继续沿用了Java 7的策略,但同时也提供了新的垃圾回收器选项。在本文中,我们将深入探讨Java 8的默认垃圾回收器以及如何查看和理解相关的JVM参数。
Java 8默认垃圾回收器
Java 8的默认垃圾回收器是Parallel Scavenge(用于新生代)和Parallel Old(用于老年代)。这种组合提供了高效的内存回收性能,尤其适合于多核处理器环境。
Parallel Scavenge
Parallel Scavenge是一个年轻代的垃圾回收器,它使用并行的复制算法来回收内存。这个收集器的主要目标是提供可预测的吞吐量,即应用程序运行时间与垃圾回收时间的比率。
Parallel Old
Parallel Old是Parallel Scavenge的老年代对应收集器,使用多线程的标记-整理算法进行内存回收。这个收集器可以与Parallel Scavenge配合使用,以实现整个堆的高效垃圾回收。
如何查看Java 8的默认垃圾回收器
要查看Java 8的默认垃圾回收器,你可以使用以下两种方法:
1. 使用命令行参数
打开命令行工具,输入以下命令:
java -XX:+PrintCommandLineFlags -version
1
这个命令会输出JVM启动时使用的命令行参数,包括垃圾回收器的配置。例如:
-XX:InitialHeapSize=534729792 -XX:MaxHeapSize=8555676672 -XX:+PrintCommandLineFlags -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:-UseLargePagesIndividualAllocation -XX:+UseParallelGC
1
在这个输出中,-XX:+UseParallelGC参数表明JVM使用了Parallel Scavenge和Parallel Old作为默认的垃圾回收器。
2. 使用Java管理扩展(JMX)API
你可以编写一个简单的Java程序来查询当前JVM使用的垃圾回收器:
import java.lang.management.GarbageCollectorMXBean;
import java.lang.management.ManagementFactory;
import java.util.List;
public class GcCollectorPrinter {
public static void main(String[] args) {
List<GarbageCollectorMXBean> beans = ManagementFactory.getGarbageCollectorMXBeans();
for (GarbageCollectorMXBean bean : beans) {
System.out.println(bean.getName());
}
}
}
运行这个程序,它会打印出当前JVM使用的垃圾回收器的名称列表。
基准方法
- 使用不同的VM参数运行相同的代码6次(
-XX:+UseSerialGC
,-XX:+UseParallelGC
,-XX:+UseConcMarkSweepGC
,-XX:ParallelCMSThreads=2
,-XX:ParallelCMSThreads=4
,-XX:+UseG1GC
) 。 - 每次运行大约需要55分钟。
- 其他VM参数:
-Xmx2048M -server
OpenJDK版本:1.8.0_51
(当前最新版本)
软体:Linux version 4.0.4-301.fc22.x86_64
硬件:Intel® Core™ i7-4790 CPU @ 3.60GHz
- 每次运行都能使用OptaPlanner解决13个计划问题。 每个计划问题要运行5分钟。 它以30秒的JVM预热开始,该预热将被丢弃。
- 解决计划问题不涉及任何IO (启动期间要加载输入的时间只有几毫秒)。 一个CPU完全饱和。 它会不断创建许多短命的对象,然后GC会收集它们。
- 基准衡量每毫秒可以计算的分数数量。 越高越好。 为拟议的规划解决方案计算分数并非易事:涉及许多计算,包括检查每个实体与每个其他实体之间的冲突。
结论
Java 8通过使用Parallel Scavenge和Parallel Old作为默认垃圾回收器,为开发者提供了一个高效且适合多核处理器的垃圾回收解决方案。通过命令行参数和JMX API,我们可以轻松地查看和验证JVM的垃圾回收器配置,从而更好地理解和优化Java应用程序的性能。