当前位置: 首页 > article >正文

JMeter Java请求开发方法

Jmeter支持多种请求方式,这里要介绍的是通过编写java实现的Java请求方式。

http请求适合大部分场景,用户能够自定义报文头、报文体、Content-Type等信息。但对于自定义拆组包,或报文带有加解密功能的情况,此时使用http请求可能就不那么容易了(可能通过beanshell也能实现,没有研究)。对于java开发工程师来说,用java的方式来编写自然是首选方案了,下面介绍如何用java来开发jmeter的java请求。

Java请求快速开始

新建maven工程

    <groupId>org.example</groupId>
    <artifactId>hello-jmeter</artifactId>
    <version>1.0-SNAPSHOT</version>

hello world

没在官网找到java请求的开发教程,直接百度了:

Jmeter 使用-JAVA请求 - 冰蓝小猪宝宝 - 博客园

参考这个,写个hello world:

添加依赖

1.在工程根目录下,创建lib目录,将apache-jmeter-5.6.3/lib/ext目录下的ApacheJMeter_core.jar文件和ApacheJMeter_java.jar文件复制到lib目录

2.在maven中添加system依赖:

        <dependency>
            <groupId>org.apache.jmeter</groupId>
            <artifactId>ApacheJMeter_core</artifactId>
            <version>1.0</version>
            <scope>system</scope>
            <systemPath>${project.basedir}/lib/ApacheJMeter_core.jar</systemPath>
        </dependency>
        <dependency>
            <groupId>org.apache.jmeter</groupId>
            <artifactId>ApacheJMeter_java</artifactId>
            <version>1.0</version>
            <scope>system</scope>
            <systemPath>${project.basedir}/lib/ApacheJMeter_java.jar</systemPath>
        </dependency>

这里的groupId和artifactId是自定义的,因为他是从本地路径读的。

编写Java测试类

编写HelloJavaSample测试类

public class HelloJavaSample extends AbstractJavaSamplerClient {
    private AtomicInteger counter = new AtomicInteger(0);
    @Override
    public void setupTest(JavaSamplerContext context) {
        super.setupTest(context);
    }

    @Override
    public void teardownTest(JavaSamplerContext context) {
        super.teardownTest(context);
    }

    public SampleResult runTest(JavaSamplerContext javaSamplerContext) {
        System.out.println("hello world!"+counter.incrementAndGet());
        SampleResult sampleResult = new SampleResult();
        sampleResult.sampleStart();
        try {
            Thread.sleep(1);
            sampleResult.setSuccessful(true);
        } catch (InterruptedException e) {
            e.printStackTrace();
            sampleResult.setSuccessful(false);
        }
        sampleResult.sampleEnd();
        sampleResult.setResponseCode("200");
        sampleResult.setResponseMessage("success!");
        sampleResult.setSamplerData("hello");
        sampleResult.setResponseData("hello world!".getBytes());
        return sampleResult;
    }
}

打包maven工程

            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-jar-plugin</artifactId>
                <version>3.0.2</version>
                <configuration>
                    <outputDirectory>${project.basedir}/result</outputDirectory>
                </configuration>
            </plugin>

修改jmeter.properties文件

修改bin/jmeter.properties文件,搜索search_paths关键字:

# List of directories (separated by ;) to search for additional JMeter plugin classes,
# for example new GUI elements and samplers.
# Any jar file in such a directory will be automatically included,
# jar files in sub directories are ignored.
# The given value is in addition to any jars found in the lib/ext directory.
# Do not use this for utility or plugin dependency jars.
#search_paths=/app1/lib;/app2/lib
search_paths=F:/my-practice/backend/hello-jmeter/result

这里的路径按自己的真实路径填写。

启动jmeter

点击右侧下拉选择,可以看到我们刚添加的HelloJavaSample类,选中。

添加必要的监听器 单笔测试

可以看到运行了1次测试,tps是1000,也能看到请求响应报文。

另外,观察控制台输出,可以看到每次都是1.也就是说每次jmeter都相当于new了一个你的java类,来进行测试的,准确的说是每个线程在整个运行过程中,使用同一个sample对象。

正式开发

以添加fastjson和hutools为例,来演示如何添加依赖包,以及添加依赖包后如何打包。

1.添加依赖

        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>fastjson</artifactId>
            <version>1.2.83</version>
        </dependency>
        <dependency>
            <groupId>cn.hutool</groupId>
            <artifactId>hutool-all</artifactId>
            <version>5.7.4</version>
        </dependency>

2.开发代码

public class HelloJsonSample extends AbstractJavaSamplerClient {
    private AtomicInteger counter = new AtomicInteger(0);
    private JSONObject template = new JSONObject();
    private String url = "http://localhost:8081/post";
    @Override
    public void setupTest(JavaSamplerContext context) {
        template.put("name", "HelloJsonSample");
        template.put("version", "1.0.0");
    }

    @Override
    public void teardownTest(JavaSamplerContext context) {
        super.teardownTest(context);
    }

    public SampleResult runTest(JavaSamplerContext javaSamplerContext) {
        int count = counter.incrementAndGet();
        template.put("count", count);
        SampleResult sampleResult = new SampleResult();
        HttpResponse httpResponse = null;
        String body = template.toJSONString();
        sampleResult.setSamplerData(body);
        try {
            sampleResult.sampleStart();
            httpResponse = HttpRequest.post(url).body(body).timeout(60000).execute();
            sampleResult.sampleEnd();
        } catch (Exception e) {
            e.printStackTrace();
            sampleResult.sampleEnd();
            sampleResult.setSuccessful(false);
            sampleResult.setResponseMessage(e.getMessage());
            return sampleResult;
        }
        String resBody = httpResponse.body();
        // 解析返回报文
        JSONObject resJson = JSON.parseObject(resBody);
        String code = resJson.getString("code");
        String retMsg = resJson.getString("retMsg");
        sampleResult.setResponseCode(code);
        sampleResult.setResponseMessage(retMsg);
        sampleResult.setResponseData(httpResponse.bodyBytes());
        sampleResult.setSentBytes(httpResponse.bodyBytes().length);
        if(!Objects.equals("000000", code)) {
            sampleResult.setSuccessful(false);
        } else {
            sampleResult.setSuccessful(true);
        }
        return sampleResult;
    }
}

这里有几点需要说明一下:

a.sampleResult.sampleStart()和sampleResult.sampleEnd()方法,不是在最外面的,而是在http调用的地方,这样统计出的响应时间更准确。

b.成功失败根据返回报文解析后的结果判断,这也符合业务需要

c.这里构造了一个template对象,这样每次迭代可以复用这个对象,公共请求信息不用每次添加。

3.复制依赖配置

将依赖也打包到${project.basedir}/result目录,同时排除掉jmeter的依赖(因为jmeter默认加载)。

            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-dependency-plugin</artifactId>
                <version>3.0.2</version>
                <executions>
                    <execution>
                        <id>copy-dependencies</id>
                        <phase>package</phase>
                        <goals>
                            <goal>copy-dependencies</goal>
                        </goals>
                        <configuration>
                            <outputDirectory>${project.basedir}/result</outputDirectory>
                            <excludeGroupIds>org.apache.jmeter</excludeGroupIds>
                        </configuration>
                    </execution>
                </executions>
            </plugin>

这段配置是为了自动复制工程依赖到result目录下,同时忽略掉jmeter那两个systemPath的依赖。

4.准备模拟服务端

用springboot启动一个简易的http服务端。

    @PostMapping("/post")
    public JSONObject post(@RequestBody Map<String, String> map){
        JSONObject json = new JSONObject();
        json.put("code", "000000");
        json.put("retMsg", "success");
        json.put("origin", map);
        log.info("接收到报文:{}", map);
        return json;
    }

修改对应的url地址。

5.打包并测试

打包前先停掉jmeter。打包结果为:

重新启动jmeter,添加新的线程组、添加java Request。

结果证明报文正确的发送到了服务端,并收到了响应。

6.开50个线程跑10秒性能

tps达到了24420,可以认为java请求的方式本身没有性能瓶颈,能够作为压测的手段。tps可以更高,因为我的模拟服务端是本地的springboot服务,往console控制台打日志了,对效率也有影响。

java请求参数

java请求参数

在java请求页面,可以看到发送参数,下面演示如何个性化设置发送参数。

添加测试类

public class HelloJsonArgSample extends AbstractJavaSamplerClient {
    private AtomicInteger counter = new AtomicInteger(0);
    private JSONObject template = new JSONObject();
    @Override
    public void setupTest(JavaSamplerContext context) {
        template.put("name", "HelloJsonSample");
        template.put("version", "1.0.0");
    }

    @Override
    public Arguments getDefaultParameters() {
        Arguments arguments = new Arguments();
        arguments.addArgument("url", "http://localhost:8081/post");
        arguments.addArgument("key1", "value1");
        return arguments;
    }

    @Override
    public void teardownTest(JavaSamplerContext context) {
        super.teardownTest(context);
    }

    public SampleResult runTest(JavaSamplerContext javaSamplerContext) {
        int count = counter.incrementAndGet();
        template.put("count", count);
        String val = javaSamplerContext.getParameter("key1");
        template.put("key1", val);
        SampleResult sampleResult = new SampleResult();
        HttpResponse httpResponse = null;
        String body = template.toJSONString();
        String url = javaSamplerContext.getParameter("url");
        sampleResult.setSamplerData(body);
        try {
            sampleResult.sampleStart();
            httpResponse = HttpRequest.post(url).body(body).timeout(60000).execute();
            sampleResult.sampleEnd();
        } catch (Exception e) {
            e.printStackTrace();
            sampleResult.sampleEnd();
            sampleResult.setSuccessful(false);
            sampleResult.setResponseMessage(e.getMessage());
            return sampleResult;
        }
        String resBody = httpResponse.body();
        // 解析返回报文
        JSONObject resJson = JSON.parseObject(resBody);
        String code = resJson.getString("code");
        String retMsg = resJson.getString("retMsg");
        sampleResult.setResponseCode(code);
        sampleResult.setResponseMessage(retMsg);
        sampleResult.setResponseData(httpResponse.bodyBytes());
        sampleResult.setSentBytes(httpResponse.bodyBytes().length);
        if(!Objects.equals("000000", code)) {
            sampleResult.setSuccessful(false);
        } else {
            sampleResult.setSuccessful(true);
        }
        return sampleResult;
    }
}

a.重载了getDefaultParameters方法

b.修改了runTest中url的获取方式

重新打包测试

关闭jmeter,重新打包,启动jmeter,添加新的线程组和java request。

可以看到,那两个参数能够在界面上看见了。

发送一笔请求,测试正常。

动态修改参数

这里的参数支持动态修改:

将value1改成value2:

但当我们切换java类时,他会自动还原。这样方便我们不修改java代码,动态调整基础参数。

CSV参数文件读取

读取csv文件测试

编写测试类

public class HelloJsonCsvSample extends AbstractJavaSamplerClient {
    private AtomicInteger counter = new AtomicInteger(0);
    private JSONObject template = new JSONObject();
    @Override
    public void setupTest(JavaSamplerContext context) {
        template.put("name", "HelloJsonSample");
        template.put("version", "1.0.0");
    }

    @Override
    public Arguments getDefaultParameters() {
        Arguments arguments = new Arguments();
        arguments.addArgument("url", "http://localhost:8081/post");
        arguments.addArgument("key1", "value1");
        return arguments;
    }

    @Override
    public void teardownTest(JavaSamplerContext context) {
        super.teardownTest(context);
    }

    public SampleResult runTest(JavaSamplerContext javaSamplerContext) {
        int count = counter.incrementAndGet();
        template.put("count", count);
        String val = javaSamplerContext.getParameter("key1");
        template.put("key1", val);
        // csv变量
        String person = javaSamplerContext.getJMeterVariables().get("person");
        String age = javaSamplerContext.getJMeterVariables().get("age");
        template.put("person", person);
        template.put("age", age);
        SampleResult sampleResult = new SampleResult();
        HttpResponse httpResponse = null;
        String body = template.toJSONString();
        String url = javaSamplerContext.getParameter("url");
        sampleResult.setSamplerData(body);
        try {
            sampleResult.sampleStart();
            httpResponse = HttpRequest.post(url).body(body).timeout(60000).execute();
            sampleResult.sampleEnd();
        } catch (Exception e) {
            e.printStackTrace();
            sampleResult.sampleEnd();
            sampleResult.setSuccessful(false);
            sampleResult.setResponseMessage(e.getMessage());
            return sampleResult;
        }
        String resBody = httpResponse.body();
        // 解析返回报文
        JSONObject resJson = JSON.parseObject(resBody);
        String code = resJson.getString("code");
        String retMsg = resJson.getString("retMsg");
        sampleResult.setResponseCode(code);
        sampleResult.setResponseMessage(retMsg);
        sampleResult.setResponseData(httpResponse.bodyBytes());
        sampleResult.setSentBytes(httpResponse.bodyBytes().length);
        if(!Objects.equals("000000", code)) {
            sampleResult.setSuccessful(false);
        } else {
            sampleResult.setSuccessful(true);
        }
        return sampleResult;
    }
}

a.在runTest方法里动态获取了参数person和age

关闭jmeter重新打包运行

一个线程跑两次:

可见读到了csv中的内容了。

扩展:分布式运行jmeter

有时候,jmeter客户端可能自身成为了一个瓶颈,比如在java请求中需要对报文做加解密等耗费CPU资源的操作,这时候可以考虑分布式运行jmeter。

官方说明:

Apache JMeter - Apache JMeter Distributed Testing Step-by-step

具体操作步骤:

为了演示方便,下面都是在本机windows上操作的,在linux上原理一样,只是把.bat后缀去掉。

1.把原jmeter复制三份

分别作为master(主控节点)、slave slave2(服务节点)

其中主控节点用于拉起压测,服务节点用于运行压测。

2.修改jmeter.properties

修改master的jmeter.properties配置,在最下方加上这几张

# 可以直接按ip和端口指定,或只指定host,端口默认就是1099了
remote_hosts=localhost:1199,localhost:1299
#server_port=1099
server.rmi.ssl.disable=true

特别关键的一步:server_port=1099这个要在master上注掉,因为我这里不想让他作为服务节点,而只是主控节点,不让他占用1099端口。

修改slave:

server_port=1199
server.rmi.ssl.disable=true
server.rmi.localport=0

指定了jmeter服务节点的端口是1199,他通过这个端口接收master发出的任务。

修改slave2:

server_port=1299
server.rmi.ssl.disable=true
server.rmi.localport=0

设置他的端口是1299。

这样就配置好了。

3.启动服务节点

双击运行slave和slave2下的命令:jmeter-server.bat 

25881、25916是他的localport,不用管。

4.启动master GUI界面并测试

运行master的jmeter.bat启动GUI界面:

以java hello world为例,运行压测(这个示例属于空跑,内部sleep1毫秒):

10线程运行10秒:

只运行1199这个服务节点试下:

这个是1199对应的服务控制台打印的日志,表示接收到了master的任务并运行了。

tps为619

再两个都启动试下:

TPS为1237,大概是619的两倍。

注意事项

1.为了测试方便,每次运行前清理上次运行结果,右侧那两个扫把样子的图标,一个是清理选中的测试结果,一个是清理所有的测试结果。一般清理所有的即可。

2.快速启用禁用组件,比如查看结果树,可能会占用点资源,那么在正式测试时,可以将其禁用。也可以鼠标右键选择enable disable。


http://www.kler.cn/a/429153.html

相关文章:

  • RV1126+FFMPEG推流项目(9)AI和AENC模块绑定,并且开启线程采集
  • Linux 管道操作
  • MySQL、HBase、ES的特点和区别
  • 代码随想录算法训练营第三十五天-动态规划-01背包(二维)
  • Red Hat8:搭建FTP服务器
  • 无降智o1 pro——一次特别的ChatGPT专业模式探索
  • Ubuntu 22.04加Windows AD域
  • 【Ubuntu】清理、压缩VirtualBox磁盘空间大小
  • 探索 Webpack:前端工程化的核心驱动力与应用场景全解析
  • 高级java每日一道面试题-2024年12月08日-JVM篇-什么是类加载器?
  • 【模型对比】ChatGPT vs Kimi vs 文心一言那个更好用?数据详细解析,找出最适合你的AI辅助工具!
  • C++运算符重载的使用——实现日期类
  • Leaflet Marker的突出显示,以及聚合
  • 医院专家抽取系统——未来之窗行业应用跨平台架构
  • Android开发-----Could not install Gradle distribution from- gradle
  • 在 Windows WSL 上部署 Ollama 和大语言模型:从镜像冗余问题看 Docker 最佳实践20241208
  • 泷羽sec学习打卡-brupsuite4
  • 高级数据结构-树状数组
  • Halcon_数据类型_ROI_仿射变换_投影变换
  • 设计模式 在SCM系统的应用场景介绍
  • ISO45001职业健康安全管理体系认证流程
  • springboot整合lua脚本在Redis实现商品库存扣减
  • 关系型数据库(RDBMS)和非关系型数据库(NoSQL)
  • 使用 Trace 实现 onnx 的导出 - 学习记录
  • golang学习,小结
  • 数学公式和科学符号在页面的展示方法