前文分享了几种性能测试中常用到的生成全局唯一标识的案例,虽然在文中我猜测了几种方案设计的性能,并根据自己的经验给出了适用的场景。
但对于一个性能测试工程师来讲,有真是测试数据才更有说服力。这让我想起来之前学过的Java微基准测试框架JMH,所以不妨一试。
JMH简介
JMH (Java Microbenchmark Harness)是一个用于编写和运行Java基准测试的工具。它被广泛用于评估Java应用程序的性能,并帮助开发人员发现和优化性能瓶颈。
JMH的主要特点包括:
高可信度:JMH提供了多种机制来消除测试过程中的噪音和偏差,确保测试结果的可靠性。
易用性:JMH提供了丰富的注解和API,使编写和运行基准测试变得相对简单。
灵活性:JMH支持多种测试模式,如简单的吞吐量测试、微基准测试以及更复杂的测试场景。
可扩展性:JMH允许用户自定义测试环境,如GC策略、编译器选项等,以满足特定的性能评估需求。
广泛应用:JMH被广泛应用于Java生态系统中,包括JDK自身的性能优化、第三方开源库的性能评估等。
JMH是Java开发者评估应用程序性能的强大工具,有助于提高Java应用程序的整体质量和性能。同样地对于性能测试而言,也可以通过JMH测试评估一段代码在实际执行当中的表现。
实测
除了使用分布式服务生成GUID这个方案以外,其他四种方案(其中两种是我自己常用的)均参与测试。原因是分布式服务需要网络交互,这个一听就不高性能,还有我暂时没条件测试这个。
下面有限展示实测结果,总结使用线程共享和线程独享的方案性能均远远高于UUID和雪花算法。为了省事儿以下测试均预热2次,预热批次大小2,测试迭代次数1次,迭代批次大小也是1次。配置如下:
.warmupIterations(2)//预热次数.warmupBatchSize(2)//预热批次大小.measurementIterations(1)//测试迭代次数.measurementBatchSize(1)//测试批次大小.build();
PS:JMH貌似还不支持Groovy所以我用Java写了这个用例。
下面是运行1个线程的测试结果:
UniqueNumberTest.exclusive thrpt 203.146 ops/usUniqueNumberTest.share thrpt 99.860 ops/usUniqueNumberTest.snow thrpt 4.096 ops/usUniqueNumberTest.uuid thrpt 11.758 ops/us
下面是运行10个线程的测试结果:
Benchmark Mode Cnt Score Error UnitsUniqueNumberTest.exclusive thrpt 1117.347 ops/usUniqueNumberTest.share thrpt 670.141 ops/usUniqueNumberTest.snow thrpt 10.925 ops/usUniqueNumberTest.uuid thrpt 3.608 ops/us
PS:此时机器的性能基本跑满了。
下面是40个线程的测试结果:
Benchmark Mode Cnt Score Error UnitsUniqueNumberTest.exclusive thrpt 1110.273 ops/usUniqueNumberTest.share thrpt 649.350 ops/usUniqueNumberTest.snow thrpt 8.908 ops/usUniqueNumberTest.uuid thrpt 4.205 ops/us
可以看出跟10个线程结果差不多。
本机配置12核心,以上的测试结果单位是微秒,把结果乘以100万就是每秒的处理量,各位在使用不同方案时可以适当参考。
测试用例
下面是我的测试用例,测试结果我就不进行可视化了。
package com.funtest.jmh;import com.funtester.utils.SnowflakeUtils;import org.openjdk.jmh.annotations.*;import org.openjdk.jmh.infra.Blackhole;import org.openjdk.jmh.results.format.ResultFormatType;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.UUID;import java.util.concurrent.TimeUnit;import java.util.concurrent.atomic.AtomicInteger;@BenchmarkMode(Mode.Throughput)//@Warmup(Ω = 3, time = 2, timeUnit = TimeUnit.SECONDS)//预热次数,含义是每个测试会跑多久//@Measurement(iterations = 3, time = 5, timeUnit = TimeUnit.SECONDS)//测试迭代次数,含义是每个测试会跑多久//@Threads(1)//测试线程数//@Fork(2)//fork表示每个测试会fork出几个进程,也就是说每个测试会跑几次@State(value = Scope.Thread)//默认为Scope.Thread,含义是每个线程都会有一个实例@OutputTimeUnit(TimeUnit.MICROSECONDS)public class UniqueNumberTest {SnowflakeUtils snowflakeUtils = new SnowflakeUtils(1, 1);ThreadLocal<Integer> exclusive = ThreadLocal.withInitial(() -> 0);AtomicInteger share = new AtomicInteger(0);@Benchmarkpublic void uuid() {UUID.randomUUID();}@Benchmarkpublic void snow() {snowflakeUtils.nextId();}@Benchmarkpublic void exclusive(Blackhole blackhole) {Integer i = exclusive.get();i++;blackhole.consume(i + "");}@Benchmarkpublic void share(Blackhole blackhole) {blackhole.consume(share.incrementAndGet() + "");}public static void main(String[] args) throws RunnerException {Options options = new OptionsBuilder().include(UniqueNumberTest.class.getSimpleName())//测试类名.result("long/result.json")//测试结果输出到result.json文件.resultFormat(ResultFormatType.JSON)//输出格式.forks(1)//fork表示每个测试会fork出几个进程,也就是说每个测试会跑几次.threads(40)//测试线程数.warmupIterations(2)//预热次数.warmupBatchSize(2)//预热批次大小.measurementIterations(1)//测试迭代次数.measurementBatchSize(1)//测试批次大小.build();new Runner(options).run();}}
感谢每一个认真阅读我文章的人,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:
这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你!有需要的小伙伴可以点击下方小卡片领取