Java:使用JMH做Benchmark基准测试

BenchMark 又叫做基准测试,主要用来测试一些方法的性能,可以根据不同的参数以不同的单位进行计算(例如可以使用吞吐量为单位,也可以使用平均时间作为单位,在 BenchmarkMode 里面进行调整)。

依赖

xml 复制代码
<dependency>
    <groupId>org.openjdk.jmh</groupId>
    <artifactId>jmh-core</artifactId>
    <version>1.19</version>
</dependency>
<dependency>
    <groupId>org.openjdk.jmh</groupId>
    <artifactId>jmh-generator-annprocess</artifactId>
    <version>1.19</version>
    <scope>provided</scope>
</dependency>

使用示例

ArrayList 和 LinkedList 的遍历的性能差别

java 复制代码
package com.demo;


import org.openjdk.jmh.annotations.*;
import org.openjdk.jmh.runner.Runner;
import org.openjdk.jmh.runner.RunnerException;
import org.openjdk.jmh.runner.options.Options;
import org.openjdk.jmh.runner.options.OptionsBuilder;

import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;
import java.util.concurrent.TimeUnit;


@BenchmarkMode(Mode.Throughput) // 吞吐量
@OutputTimeUnit(TimeUnit.MILLISECONDS) // 结果所使用的时间单位
@State(Scope.Thread) // 每个测试线程分配一个实例
@Fork(2) // Fork进行的数目
@Warmup(iterations = 4) // 先预热4轮
@Measurement(iterations = 10) // 进行10轮测试
public class BenchMarkDemo {

    @Param({"10", "40", "70", "100"}) // 定义四个参数,之后会分别对这四个参数进行测试
    private int n;

    private List<Integer> array;
    private List<Integer> list;

    @Setup(Level.Trial) // 初始化方法,在全部Benchmark运行之前进行
    public void init() {
        array = new ArrayList<>(0);
        list = new LinkedList<>();
        for (int i = 0; i < n; i++) {
            array.add(i);
            list.add(i);
        }
    }

    @Benchmark
    public void arrayTraverse() {
        for (int i = 0; i < n; i++) {
            array.get(i);
        }
    }

    @Benchmark
    public void listTraverse() {
        for (int i = 0; i < n; i++) {
            list.get(i);
        }
    }

    @TearDown(Level.Trial) // 结束方法,在全部Benchmark运行之后进行
    public void arrayRemove() {
        for (int i = 0; i < n; i++) {
            array.remove(0);
            list.remove(0);
        }
    }

    public static void main(String[] args) throws RunnerException {
        Options options = new OptionsBuilder().include(BenchMarkDemo.class.getSimpleName()).build();
        new Runner(options).run();
    }
}
复制代码
# Run complete. Total time: 00:03:56

Benchmark                    (n)   Mode  Cnt       Score       Error   Units
BenchMarkDemo.arrayTraverse   10  thrpt   20  337852.300 ± 10230.514  ops/ms
BenchMarkDemo.arrayTraverse   40  thrpt   20  342619.598 ±  6272.177  ops/ms
BenchMarkDemo.arrayTraverse   70  thrpt   20  342534.411 ±  6018.479  ops/ms
BenchMarkDemo.arrayTraverse  100  thrpt   20  333470.068 ± 17285.845  ops/ms
BenchMarkDemo.listTraverse    10  thrpt   20   45899.695 ±  2960.264  ops/ms
BenchMarkDemo.listTraverse    40  thrpt   20    6615.649 ±   201.404  ops/ms
BenchMarkDemo.listTraverse    70  thrpt   20    1910.175 ±    31.484  ops/ms
BenchMarkDemo.listTraverse   100  thrpt   20     787.424 ±    25.747  ops/ms

可以结合 Score 和 Unit 这两列,看到方法的效率。这里显然 arrayTraverse 的效率比 listTraverse 的高很多,因为 Unit 单位是 ops/ms,即单位时间内执行的操作数。所以显然在遍历的时候,ArrayList的效率是比LinkedList高的。

参考
使用JMH做Benchmark基准测试

相关推荐
Javatutouhouduan3 小时前
2026Java面试的正确打开方式!
java·高并发·java面试·java面试题·后端开发·java编程·java八股文
chao1898444 小时前
基于 SPEA2 的多目标优化算法 MATLAB 实现
开发语言·算法·matlab
JAVA面经实录9174 小时前
Java初级最终完整版学习路线图
java·spring·eclipse·maven
赏金术士4 小时前
Kotlin 习题集 · 高级篇
android·开发语言·kotlin
原来是猿5 小时前
网络计算器:理解序列化与反序列化(中)
linux·运维·服务器·网络·tcp/ip
Cat_Rocky5 小时前
k8s-持久化存储,粗浅学习
java·学习·kubernetes
楼兰公子5 小时前
buildroot 在编译rust时裁剪平台类型数量的方法
开发语言·后端·rust
知识领航员5 小时前
蘑兔AI音乐深度实测:功能拆解、实测表现与适用场景
java·c语言·c++·人工智能·python·算法·github
AOwhisky5 小时前
虚拟化技术学习笔记
linux·运维·笔记·学习·虚拟化技术
吴声子夜歌5 小时前
Go——并发编程
开发语言·后端·golang