[高性能 Java] 1BRC 比赛优秀代码分析
The One Billion Row Challenge(1BRC): Java 处理亿级别数据的性能调优 这篇文章介绍了 1BRC 比赛,以及笔者个人思考的一些可能的优化。
本次获奖的第一名大佬 @thomaswue 是 GraalVM 项目的 lead,第三名大佬 @jerrinot 是 QuestDB 成员之一。
本文将介绍比赛代码中都用了哪些比较有意思的优化。比赛前三名用到的优化大同小异,后续介绍以第一名代码(No.1代码)为例。
优化一: AOT 编译
使用了 21.0.2-graal。
使用 GraalVM 可以 静态编译 Java 字节码(bytecode, .class 文件) ,得到二进制机器码 (native binary),提升 Java 程序性能。
Java的执行过程可以分为两个部分:
- javac 将 Java 源文件编译成 Java 字节码;
- jvm 将字节码逐条解释执行。
在解释执行的过程中,如果发现热点代码,使用即时编译器(JIT,更多细节参考Java及时编译器)将其直接编译成机器码;下次执行时直接运行机器码,这样的速度比解释执行要快。在JIT编译器将字节码翻译成机器码之前的运行,被称作 warmup,jvm 在这个过程中发现热点代码。
GraalVM 支持通过静态编译将Java字节码翻译成机器码,这样的好处包括:可执行文件更小;由于无需 warmup,运行的直接就是机器码,能更快的达到峰值性能。
优化二: 系统层面优化
先 show 代码
int numberOfWorkers = Runtime.getRuntime().availableProcessors();
try (var fileChannel = FileChannel.open(java.nio.file.Path.of(FILE), java.nio.file.StandardOpenOption.READ)) {
long fileSize = fileChannel.size();
final long fileStart = fileChannel.map(FileChannel.MapMode.READ_ONLY, 0, fileSize, java.lang.foreign.Arena.global()).address();
final long fileEnd = fileStart + fileSize;
final AtomicLong cursor = new AtomicLong(fileStart);
// Parallel processing of segments.
Thread[] threads = new Thread[numberOfWorkers];
List<Result>[] allResults = new List[numberOfWorkers];
for (int i = 0; i < threads.length; ++i) {
final int index = i;
threads[i] = new Thread(() -> {
List<Result> results = new ArrayList<>(MAX_CITIES);
parseLoop(cursor, fileEnd, fileStart, results);
allResults[index] = results;
});
threads[i].start();
}
- 多线程优化:拿到 JVM 能够获取的所有processor,启动相同数目的线程处理数据。提高 CPU 利用率。
- mmap + unsafe:
FileChannel open(Path path, OpenOption... options)
和MappedByteBuffer map(MapMode mode, long position, long size)
将输入文件 mmap 到内存中,获取内存地址后,利用 unsafe 逐个字节的读取处理(而不是直接使用的 Java 中 FileReader)。减少用户、内核态切换,cpu 友好。
前面代码中可以看到,每个线程调用 parseLoop
函数来进行处理。
在parseLoop
函数中,每次读一个 batch 的数据(SEGMENT_SIZE 大小),进一步将 batch 均分成 3 份,然后一个线程中同时处理这三份数据。代码如下所示。
private static void parseLoop(AtomicLong counter, long fileEnd, long fileStart, List<Result> collectedResults) {
Result[] results = new Result[HASH_TABLE_SIZE];
while (true) {
long current = counter.addAndGet(SEGMENT_SIZE) - SEGMENT_SIZE;
if (current >= fileEnd) {
return;
}
/** 处理边界 **/
long dist = (segmentEnd - segmentStart) / 3;
long midPoint1 = nextNewLine(segmentStart + dist);
long midPoint2 = nextNewLine(segmentStart + dist + dist);
Scanner scanner1 = new Scanner(segmentStart, midPoint1);
Scanner scanner2 = new Scanner(midPoint1 + 1, midPoint2);
Scanner scanner3 = new Scanner(midPoint2 + 1, segmentEnd);
while (true) {
if (!scanner1.hasNext()) {
break;
}
if (!scanner2.hasNext()) {
break;
}
if (!scanner3.hasNext()) {
break;
}
/** 同时迭代 scanner1、scanner2、scanner3 **/
}
while (scanner1.hasNext()) {
/** 迭代剩余 scanner1 **/
}
while (scanner2.hasNext()) {
/** 迭代剩余 scanner2 **/
}
while (scanner3.hasNext()) {
/** 迭代剩余 scanner3 **/
}
}
}
batch 读取后,切分成3份并发处理,优化思路如下:
- batch 读能利用缓存,降低 IO 开销,避免更小粒度读取
- 并发处理,提高 cpu 利用率
- 加快 JVM 识别、优化热点代码
优化三: partition 策略,work steal 优于 equal split
并行读取大文件,最直观、简单的方法是:先将文件均匀切分(equal split),然后每个线程读取属于各自的那部分数据。但这样做也可能存在劣势,即某个线程处理完自己那份数据后就开始“摸鱼”了。
第一名则采用了类似 "work stealing" 的策略。使用一个原子变量表示文件指针,每个线程尝试读取文件最新的那个 batch 数据(2MB)。核心代码是
long current = counter.addAndGet(SEGMENT_SIZE) - SEGMENT_SIZE;
。这样就避免了有活没干完但还有线程“摸鱼”的情况。
private static void parseLoop(AtomicLong counter, long fileEnd, long fileStart, List<Result> collectedResults) {
Result[] results = new Result[HASH_TABLE_SIZE];
while (true) {
long current = counter.addAndGet(SEGMENT_SIZE) - SEGMENT_SIZE;
if (current >= fileEnd) {
return;
}
long segmentEnd = nextNewLine(Math.min(fileEnd - 1, current + SEGMENT_SIZE));
long segmentStart;
if (current == fileStart) {
segmentStart = current;
}
else {
segmentStart = nextNewLine(current) + 1;
}
// ...
}
// ...
}
优化四: lookup table 数据结构
此处利用 mask 方法实现了一个 lookup table 数据结果(没有直接用 hashset)。主要逻辑在 findResult函数 中。 主要是精巧的按位运算,暂时没有时间深入看,感兴趣的朋友可以自行看下。
QuestDB 参赛员工的优化思路
有位 QuestDB 的参赛员工(@Marko Topolnik)也写了篇博客The Billion Row Challenge (1BRC) - Step-by-step from 71s to 1.7s,介绍了他参赛时的优化思路。后面会快速介绍一下前面没介绍的优化。
语言内置脚手架写一个可用版本
只要 17 行,写得十分简洁,也兼顾了性能 * 用到了并行 * 避免使用耗时的正则
使用 OpenJDK 21,最终耗时 71s,比第一名慢了 47 倍。
var allStats = new BufferedReader(new FileReader("measurements.txt"))
.lines()
.parallel()
.collect(
groupingBy(line -> line.substring(0, line.indexOf(';')),
summarizingDouble(line ->
parseDouble(line.substring(line.indexOf(';') + 1)))));
var result = allStats.entrySet().stream().collect(Collectors.toMap(
Entry::getKey,
e -> {
var stats = e.getValue();
return String.format("%.1f/%.1f/%.1f",
stats.getMin(), stats.getAverage(), stats.getMax());
},
(l, r) -> r,
TreeMap::new));
System.out.println(result);
换一个 JVM
换了一个 JVM,使用 GraalVM,将源码编译成二进制程序、而非字节码,减少 JVM 启动开销。 程序从 71s -> 66s,有 7.5% 提升。
先 profiling,然后 optimize
这也很重要的系统性能优化的方法论之一,通过 profile 发现瓶颈,然后针对性优化。
作者用到了三个 profiling 工具:
- VisualVM:作者主要用了 VisualGC plugin, 支持 100ms 采样,能实时显示 JIT编译和 GC 情况。
- 还支持 Java 程序的 cpu profiling、memory profiling、 monitor、线程分析、thread dump 等其他功能。
- async-profiler:支持 cpu、heap 的采样分析,结合 jbang 能够画火焰图。
- linux-perf-cmd: 作者主要获取 底层 cpu 的计数器信息, 用到的命令是
perf stat
,perf stat -e branches,branch-misses,cache-references,cache-misses,cycles,instructions,idle-cycles-backend,idle-cycles-frontend,task-clock -- java --enable-preview -cp src Blog11
- 系统级别的 perf,命令行工具,包含kernel、硬件等级别的 profile 数据,比如执行了多少指令、访存多少次、多少次分支预测、多少次 分支预测失败、多少次 cache miss 等等。
作者使用工具的顺序如下: * VisualGC: 优化内存分配、GC 相关问题 * Async-Profiler: 通过火焰图,找到代码中的热点 * perf stat:当程序运行时间到 3s 左右时,开始指令分析。
作者原文分析非常有意思,如下:
VisualGC was the most useful in the initial optimization phase. Then, once I sorted out allocation, the flamegraph proved highly useful to pinpoint the bottlenecks in the code. However, once the runtime went below ~3 seconds, its usefulness declined. At this level we're squeezing out performance not from methods, but from individual CPU instructions. This is where perf stat became the best tool.
For reference, here's the perf stat report for our basic implementation:
数据 metric 393,418,510,508 branches 3,112,052,890 branch-misses 26,847,457,554 cache-references 982,409,158 cache-misses 756,818,510,991 cycles 2,031,528,945,161 instructions It's most helpful to interpret the numbers on a per-row basis (dividing everything by 1 billion). We can see that the program spends more than 2,000 instructions on each row. No need to get into more details; initially we'll be driving down just this metric.
从 MemorySegment 到 sun.misc.Unsafe
MemorySegment 是 jdk 17 的 incubator 特性,支持连续内存映射,可以用于实现 mmap。
虽然它能带来很大性能提升,但是相比 sum.misc.Unsafe
有更多的 bounds check 开销。
位运算替代 if,减少 branch miss
if-else 对应的 cpu 指令运行时涉及到 分支预测(现代 cpu都支持指令预取、执行)。一次分支预测失败的开销是 10~15 个指令。
在切分数据的时候,需要以分号来切分。为了避免 branch miss 带来的额外开销,作者使用精巧的位运算实现替代 if-else。
SWAR 优化
SWAR (SIMD Within A Register) . 在解析数据的时候,一次只读 8 byte(存放在 long 中)。每次对 long 型数据进行 位运算, 寻找分号作为分隔符。 8 byte的数据正好放在一个寄存器里面,每次位运算相当于对 8 byte 的数据做 SMID。 减少了 cpu 指令数。
通过该优化,每条数据所消耗的指令数下降到原来的 1/3.
优化指令分支预测
写了个模拟解析的程序,拿到指令中热点代码的 cpu 分支预测表 (Branch history table, BHT)。通过循环展开(unroll loop), 减少分支预测。
Eliminate startup/cleanup
由于清理 mmap 中 unmapping memory-mapped file 需要一定时间,一种比较 tricky 的做法是起一个 subprocess,异步清理。 当主程序计算完毕之后即可退出,subprocess 在后台清理,这样比赛检测程序就不会计时清理时间。
优化之外
性能优化应当避免以牺牲最佳实践为代价。生产实践中的代码需要有较高的可读性/可维护性、鲁棒性。 因为用户行为总是非预期的,所以 validation、bounds check、错误恢复、hash resizing 等都是必须的。