1、JMH简介

JMHJava Microbenchmark Harness,是Java用来做基准测试的一个工具,该工具由OpenJDK提供并维护,测试结果可信度高。

相对于 Jmeter、ab ,它通过编写代码的方式进行压测,在特定场景下会更能评估某项性能。

本次通过使用JMH来压测Dubbo的性能(官方也是使用JMH压测)

2、使用

只需要引用两个jar即可:

<dependency><groupId>org.openjdk.jmh</groupId><artifactId>jmh-core</artifactId><version>1.29</version>
</dependency>
<dependency><groupId>org.openjdk.jmh</groupId><artifactId>jmh-generator-annprocess</artifactId><version>1.29</version>
</dependency>

通过一系列的注解即可使用JMH。

@State

只能用在类上,有三个取值:

Scope.Thread:默认的State,每个测试线程分配一个实例;
Scope.Benchmark:所有测试线程共享一个实例,用于测试有状态实例在多线程共享下的性能;
Scope.Group:每个线程组共享一个实例;

@OutputTimeUnit

时间单位,如毫秒 TimeUnit.MILLISECONDS、秒 TimeUnit.SECONDS

@Benchmark

声明一个public方法为基准测试方法。该类下的所有被@Benchmark注解的方法都会执行。

相当于类的main方法

@BenchmarkMode

指定测试某个接口的指标,如吞吐量、平均执行时间,一般我都是选择 ALL

Mode有:

  • Throughput: 整体吞吐量,例如“1秒内可以执行多少次调用” (thrpt,参加第5点)

  • AverageTime: 调用的平均时间,例如“每次调用平均耗时xxx毫秒”。(avgt)

  • SampleTime: 随机取样,最后输出取样结果的分布,例如“99%的调用在xxx毫秒以内,99.99%的调用在xxx毫秒以内”(simple)

  • SingleShotTime: 以上模式都是默认一次 iteration 是 1s,唯有 SingleShotTime 是只运行一次。往往同时把 warmup 次数设为0,用于测试冷启动时的性能。(ss)

@BenchmarkMode({Mode.Throughput,Mode.All})
public class StressTestProvider {}

@Measurement

用于控制压测的次数

//测量2次,每次测量的持续时间为20秒
@Measurement(iterations = 2, time = 20 , timeUnit = TimeUnit.SECONDS)

@Warmup

预热,预热可以避免首次因为一些其他因素,如CPU波动、类加载耗时这些情况的影响。

@Warmup(iterations = 3, time = 1, timeUnit = TimeUnit.SECONDS)

参数解释同上。

@Fork

@Fork用于指定fork出多少个子进程来执行同一基准测试方法。

@Threads

@Threads注解用于指定使用多少个线程来执行基准测试方法,如果使用@Threads指定线程数为2,那么每次测量都会创建两个线程来执行基准测试方法。

3、运行

我这里的例子是压测dubbo,源码链接在文末

完整例子:

@BenchmarkMode({Mode.All})
@Warmup(iterations = 3, time = 5, timeUnit = TimeUnit.SECONDS)
//测量次数,每次测量的持续时间
@Measurement(iterations = 3, time = 10, timeUnit = TimeUnit.SECONDS)
@Threads(32)
@Fork(1)
@State(Scope.Benchmark)
@OutputTimeUnit(TimeUnit.SECONDS)
@Slf4j
public class StressTestProvider {private final AnnotationConfigApplicationContext annotationConfigApplicationContext;private final StressTestController stressTestController;public StressTestProvider() {annotationConfigApplicationContext = new AnnotationConfigApplicationContext(AnnotationConfig.class);annotationConfigApplicationContext.start();stressTestController = annotationConfigApplicationContext.getBean("stressTestController", StressTestController.class);}@TearDownpublic void close() throws IOException {annotationConfigApplicationContext.close();}@Benchmarkpublic void string1k() {stressTestController.string1k();}@Benchmarkpublic void string100k() {stressTestController.string100k();}public static void main(String[] args) throws RunnerException {log.info("测试开始");Options opt = new OptionsBuilder().include(StressTestProvider.class.getSimpleName())//可以通过注解注入
//                .warmupIterations(3)
//                .warmupTime(TimeValue.seconds(10))//报告输出.result("result.json")//报告格式.resultFormat(ResultFormatType.JSON).build();new Runner(opt).run();}
}

有两种运行的方式,一般采用打成jar这种。

3.1、main方法运行

如上,只需要 配置Options,运行main方法即可,注意要使用 run模式启动,不要使用debug模式启动。

否则会报错:

transport error 202: connect failed: Connection refused ERROR

3.2、打成jar运行

有时候需要放在服务器上运行,就需要打成一个jar,需要使用单独的jar打包插件:

<build><plugins><plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-shade-plugin</artifactId><version>2.2</version><executions><execution><phase>package</phase><goals><goal>shade</goal></goals><configuration><finalName>jmh-demo</finalName><transformers><transformerimplementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer"><mainClass>org.openjdk.jmh.Main</mainClass></transformer><transformerimplementation="org.apache.maven.plugins.shade.resource.ServicesResourceTransformer"/></transformers></configuration></execution></executions></plugin></plugins>
</build>

如果不想要这种打包方式,打成jar的时候一定要声明main方法入口对应的类,也就是上面StressTestProvider

还有就是,因为我的是springboot项目,我测试了一下想同时打包springboot和 jmh:

但是运行 jhm-demo.jar 发现报错:not match main class,还是老老实实通过 profile 节点打包吧。

打完包后,通过以下命令即可运行:

java -jar jmh-demo.jar  -rf json -rff result.json

-rf json 是输出 json的格式

-rff /data/result.json 是输出文件位置和名称

4、结果

执行后,会生成一个汇总结果:

Result "com.dubbo.benchmark.StressTestProvider.string1k":N = 3mean =      0.016 ±(99.9%) 0.022 s/opHistogram, s/op:[0.014, 0.014) = 0 [0.014, 0.015) = 0 [0.015, 0.015) = 0 [0.015, 0.015) = 1 [0.015, 0.015) = 1 [0.015, 0.016) = 0 [0.016, 0.016) = 0 [0.016, 0.016) = 0 [0.016, 0.016) = 0 [0.016, 0.017) = 0 [0.017, 0.017) = 0 [0.017, 0.017) = 0 [0.017, 0.017) = 1 [0.017, 0.018) = 0 [0.018, 0.018) = 0 [0.018, 0.018) = 0 Percentiles, s/op:p(0.0000) =      0.015 s/opp(50.0000) =      0.015 s/opp(90.0000) =      0.017 s/opp(95.0000) =      0.017 s/opp(99.0000) =      0.017 s/opp(99.9000) =      0.017 s/opp(99.9900) =      0.017 s/opp(99.9990) =      0.017 s/opp(99.9999) =      0.017 s/opp(100.0000) =      0.017 s/op# 第36行
# Run complete. Total time: 00:05:12Benchmark                                           Mode     Cnt     Score      Error  Units
StressTestProvider.string100k                      thrpt       3   759.794 ±   66.300  ops/s
StressTestProvider.string1k                        thrpt       3  6798.005 ± 6992.093  ops/s
StressTestProvider.string100k                       avgt       3     0.042 ±    0.002   s/op
StressTestProvider.string1k                         avgt       3     0.005 ±    0.012   s/op
StressTestProvider.string100k                     sample   22982     0.042 ±    0.001   s/op
StressTestProvider.string100k:string100k·p0.00    sample             0.017              s/op
StressTestProvider.string100k:string100k·p0.50    sample             0.041              s/op
StressTestProvider.string100k:string100k·p0.90    sample             0.048              s/op
StressTestProvider.string100k:string100k·p0.95    sample             0.050              s/op
StressTestProvider.string100k:string100k·p0.99    sample             0.058              s/op
StressTestProvider.string100k:string100k·p0.999   sample             0.075              s/op
StressTestProvider.string100k:string100k·p0.9999  sample             0.088              s/op
StressTestProvider.string100k:string100k·p1.00    sample             0.092              s/opStressTestProvider.string1k                       sample  186906     0.005 ±    0.001   s/op
StressTestProvider.string1k:string1k·p0.00        sample             0.001              s/op
StressTestProvider.string1k:string1k·p0.50        sample             0.005              s/op
StressTestProvider.string1k:string1k·p0.90        sample             0.007              s/op
StressTestProvider.string1k:string1k·p0.95        sample             0.008              s/op
StressTestProvider.string1k:string1k·p0.99        sample             0.011              s/op
StressTestProvider.string1k:string1k·p0.999       sample             0.030              s/op
StressTestProvider.string1k:string1k·p0.9999      sample             0.035              s/op
StressTestProvider.string1k:string1k·p1.00        sample             0.038              s/op
StressTestProvider.string100k                         ss       3     0.030 ±    0.181   s/op
StressTestProvider.string1k                           ss       3     0.016 ±    0.022   s/opBenchmark result is saved to result.json

结果分析

简单分析一下:

只需要从第36行开始看,我这里一共压测了2个方法

  • StressTestProvider.string100k
  • StressTestProvider.string1k

Mode

这一列表示测试的名称,也就是 @BenchmarkMode你选择的测试类型,源码在此:

public enum Mode {/*** <p>Throughput: operations per unit of time.</p>*/Throughput("thrpt", "Throughput, ops/time"),/*** <p>Average time: average time per per operation.</p>**/AverageTime("avgt", "Average time, time/op"),/*** <p>Sample time: samples the time for each operation.</p>**/SampleTime("sample", "Sampling time"),/*** <p>Single shot time: measures the time for a single operation.</p>**/SingleShotTime("ss", "Single shot invocation time"),

thrpt:吞吐量,也可以理解为tps、ops

avgt:每次请求的平均耗时

sample:请求样本数量,这次压测一共发了多少个请求

ss:除去冷启动,一共执行了多少轮

Cnt、Score、Units

单位

Error

误差

如果你配置了输出文件,比如我上面的 resul.json ,但是你打开是看不懂的,可以借助两个网站把文件上传进行分析:

  • https://jmh.morethan.io
  • http://deepoove.com/jmh-visual-chart

汇总:

以上对dubbo进行了分别传输1k和100k的数据压测。

provider机器:

2核4gCentOS release 6.4 (Final)
model name      : QEMU Virtual CPU version 2.5+
stepping        : 3
cpu MHz         : 2099.998
cache size      : 4096 KB

JVM:

jdk1.8
-server -Xmx2g -Xms2g -XX:+UseG1GC

dubbo:

版本:2.7.3
序列化:hessian2
使用默认dubbo线程数

压测参数:

32并发

1k 100k
TPS 6700 760
RTT 95% 8ms 95% 50ms
AVGTime/OP 5ms 42ms
OOM

对比了 jmeter、Apache-Benmark(ab)、jmh 这三个压测工具,个人比较推荐使用jmh,原因有:

  • jmh压测简单,只需要引入依赖,声明注解
  • 准确性高,目前大多数性能压测都是使用jmh
  • 缺点就是代码入侵

灵感参考:

  • dubbo压测的官方代码:https://github.com/apache/dubbo-benchmark
  • 主流RPC框架压测代码:https://github.com/hank-whu/rpc-benchmark

  • 压测dubbo的源码已上传到github:https://github.com/DogerRain/dubbo-samples-test

分享一些Java资源:

Java学习路线思维导图+Java学习视频+简历模板+Java电子书

(九)JMH的详细使用,以及压测dubbo相关推荐

  1. jmeter压测dubbo详解

    一.Dubbo简介 dubbo是一个分布式服务框架,致力于提供高性能和透明化的RPC远程服务调用方案,以及SOA服务治理方案. 二.Dubbo接口测试方案 方法:使用jmeter测试dubbo接口 1 ...

  2. 字节跳动全链路压测(Rhino)的实践

    1. 背景 随着公司业务的不断扩张,用户流量在不断提升,研发体系的规模和复杂性也随之增加.线上服务的稳定性也越来越重要,服务性能问题,以及容量问题也越发明显. 因此有必要搭建一个有效压测系统,提供安全 ...

  3. 春节保卫战:腾讯百万 QPS 线上环境云压测方案解析

    导语|春节期间腾讯大部分业务进入流量备战的紧张时刻.压测相比于监控而言,是更具主动性的筹备手段.通过高负载.真实流量的预演,探测系统的瓶颈和发现风险,是服务质量保障体系的重要一环.云压测主要聚焦在压测 ...

  4. 高德全链路压测平台TestPG的架构与实践

    导读 2018年十一当天,高德DAU突破一个亿,不断增长的日活带来喜悦的同时,也给支撑高德业务的技术人带来了挑战.如何保障系统的稳定性,如何保证系统能持续的为用户提供可靠的服务?是所有高德技术人面临的 ...

  5. 高德地图全链路压测平台TestPG的架构与实践

    高德地图:全链路压测平台TestPG的架构与实践 转自  https://www.sohu.com/a/341414025_692515 1. 导读 2019年以来,高德DAU一个亿进入常态,不断增长 ...

  6. 多线程与高并发(九):单机压测工具JMH,单机最快MQ - Disruptor原理解析

    单机压测工具JMH JMH Java准测试工具套件 什么是JMH 官网 http://openjdk.java.net/projects/code-tools/jmh/ 创建JMH测试 1.创建Mav ...

  7. mysql为什么要压测_mysql集群压测的详细介绍

    本篇文章给大家带来的内容是关于mysql集群压测的详细介绍,有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助. mysql压测 mysql自带就有一个叫mysqlslap的压力测试工具,通 ...

  8. Jmeter阶梯式等值压测-详细实战教程(一)

    Jmeter阶梯式等值压测-详细实战教程 导语:全能测开之路小白–Jmeter 作者:变优秀的小白 QQ学习交流群(new): 811792998 QQ群(new): 811792998 注:如中途遇 ...

  9. 大厂钟爱的全链路压测有什么意义?四种压测方案详细对比分析

    全链路压测? 基于实际的生产业务场景和系统环境,模拟海量的用户请求和数据,对整个业务链路进行各种场景的测试验证,持续发现并进行瓶颈调优,保障系统稳定性的一个技术工程. 针对业务场景越发复杂化.海量数据 ...

最新文章

  1. 深度学习LiDAR定位:L3-Net
  2. BERT各个场景实例代码
  3. Windows Socket五种I/O模型——代码全攻略
  4. java代码写jsp读取,Java IO学习基础之读写文本文件-JSP教程,Java技巧及代码
  5. 重写HttpServlet
  6. 写在前面-Terraform
  7. weblogic配置domain和删除domain
  8. 从Windows到Linux
  9. php给别人写接口,php给客户端写接口记录
  10. 认知空间是什么意思_为什么会选择3D立体摄影
  11. 碱度控制化学品行业调研报告 - 市场现状分析与发展前景预测
  12. mysql 5.622_新特新解读 | MySQL 8.0 对 count(*)的优化
  13. centos7搭建radius认证服务器
  14. 小米蓝牙音响驱动_小米手机绝佳配件,20W无线闪充,支持唤醒小爱,只卖149
  15. java wed汽车4s店销售管理系统
  16. 51单片机系列——定时/计数器
  17. NEYC 1702 排座 问题模型
  18. android如何基于父布局,如何根据父视图的尺寸调整Android视图的大小
  19. 计算机机房年度重点工作,信息中心计算机的机房建设要求最新.doc
  20. Aeon项目今天正式启动

热门文章

  1. 图卷积神经网络(GCN)目前的应用领域
  2. 【系统辨识】最小二乘法
  3. 【贝叶斯神经网络训练】(torch实现)
  4. centos7 搭建 svn 服务端
  5. java 日历工具_java之日历处理工具类Calendar类
  6. 安装LINUX操作系统(CENT OS)
  7. 软件测试人员去外包公司待遇怎么样?外包薪资高吗?
  8. FX5UPLC的各种以太网通讯,SOCKET 2台PLC简单通讯 走网线
  9. 贝叶斯软件genle教程_贝叶斯网络软件SMILE和GENIE的使用
  10. vue3实现吸顶效果