Jmeter的Sampler用来模拟向被测试系统发起请求的,Jmeter会记录取样的结果,也就是TPS、响应时间等性能指标。接下来就是使用Java请求进行一个接口测试。
大致流程:
1、导入jar包:导入Jmeter相关jar包
2、编写脚本:使用Eclipse或者IntelliJ IDEA,进行Java测试脚本的编写
3、导出为jar包:使用Eclipse或者IntelliJ IDEA将测试脚本编译打包为jar包导出
4、配置Jmeter:使用自编译的jar包,以及一些相关配置进行性能测试
5、选择自编写jar包:自编译jar包
6、Jmeter运行:开始运行
实战操作
一、核心步骤(以我实际操作为例)
1.创建工程
使用IntelliJ IDEA创建一个Java工程,取名为JmeterTest;
2.添加依赖
2.1lib及子目录依赖
a.使用Jmeter运行Java脚本,需要用到Jmeter提供的框架jar包(分别在Jmeter目录下的lib和ext目录下)
ext:
ApacheJMeter_core.jar
ApacheJMeter_java.jar
lib:
slf4j-api-1.7.25.jar
jorphan.jar
Jmeter5.3需要多添加一个jar包:oro-2.0.8.jar
b.将JMeter的lib目录下的jar文件添加进此工程;
注意:此处有坑,不能只添加lib这个大目录,还需要添加lib目录下的ext和junit目录,否则无法调用相关jar包
lib目录下两个两个子目录
2.2IDEA操作
1、打开 File -> Project Structure (Ctrl + Shift + Alt + S)或者使用快捷键
3.脚本编写
创建一个类并实现JavaSamplerClient接口或继承AbstractJavaSamplerClient,并重写:
public Arguments getDefaultParameters():设置可用参数及的默认值;自定义的参数,它可以获取到Jmeter面板上的参数
public void setupTest(JavaSamplerContext arg0):每个线程测试前执行一次,做一些初始化工作;
public SampleResult runTest(JavaSamplerContext arg0):开始测试,从arg0参数可以获得参数值;执行业务代码函数,在setupTest方法后执行,每个线程执行N次
public void teardownTest(JavaSamplerContext arg0):测试结束时调用;
query:
package cn.testfan;
import java.sql.*;
public class queryTest {
public static void main(String[] args) throws ClassNotFoundException, SQLException {
// 1.注册驱动
Class.forName("com.mysql.cj.jdbc.Driver");
String url = "jdbc:mysql://localhost:3306/test?useUnicode=true&characterEncoding=utf8";
String userName = "root";
String password = "root";
String sql = "SELECT * FROM boys";
// 2.建立链接:通过ip、端口号、数据库名、用户名、密码
Connection conn = DriverManager.getConnection(url,userName,password);
// 3.对SQL语句做预编译,提升执行性能,返回一个经过变异后的SQL语句对象PreparedStatement
PreparedStatement ps = conn.prepareStatement(sql);
// 5.执行SQL
ResultSet set = ps.executeQuery();
while (set.next()){
String boyNum = set.getString("boyNum");
String matchNum = set.getString("matchNum");
System.out.println("boyNum为:" + boyNum);
System.out.println("matchNum为:" + matchNum);
System.out.println("++++++++++");
}
// 6.关闭数据库链接
conn.close();
}
}
转换为Jmeter可以执行的脚本:
类名后添加implements JavaSamplerClient,点击ALT+Enter
update:
package cn.testfan;
import org.apache.jmeter.config.Arguments;
import org.apache.jmeter.protocol.java.sampler.JavaSamplerClient;
import org.apache.jmeter.protocol.java.sampler.JavaSamplerContext;
import org.apache.jmeter.samplers.SampleResult;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.PreparedStatement;
import java.sql.SQLException;
/*
自定义的测试类,实现了Jmeter的接口JavaSamplerClient
*/
public class InsertTest implements JavaSamplerClient {
// String url = "jdbc:mysql://localhost:3306/test?useUnicode=true&characterEncoding=utf8";
// String userName = "root";
// String password = "root";
// String boyNum = "boy-001";
// String matchNum = "111";
String sql = "INSERT INTO boys (boyNum, matchNum) VALUES (?,?)";
PreparedStatement ps;
Connection conn;
public static void main(String[] args) {
// 模拟Jmeter执行一次脚本
InsertTest test = new InsertTest();
JavaSamplerContext context = new JavaSamplerContext(test.getDefaultParameters());
test.setupTest(context);
test.runTest(context);
test.teardownTest(context);
}
/*
* 初始化函数,每个线程先执行此函数,并且仅执行一次
*/
@Override
public void setupTest(JavaSamplerContext javaSamplerContext) {
try {
// 1.注册驱动
Class.forName("com.mysql.cj.jdbc.Driver");
// 2.建立链接:通过ip、端口号、数据库名、用户名、密码
String dbUrl = javaSamplerContext.getParameter("dbUrl");
String dbUserName = javaSamplerContext.getParameter("dbUserName");
String dbPassWord = javaSamplerContext.getParameter("dbPassWord");
conn = DriverManager.getConnection(dbUrl,dbUserName,dbPassWord);
// 3.对SQL语句做预编译,提升执行性能,返回一个经过变异后的SQL语句对象PreparedStatement
ps = conn.prepareStatement(sql);
} catch (ClassNotFoundException | SQLException e) {
e.printStackTrace();
}
}
/*
* 执行业务代码函数,在setupTest方法后执行,每个线程执行N次
*/
@Override
public SampleResult runTest(JavaSamplerContext javaSamplerContext) {
//给请求起个名字
SampleResult sampleResult = new SampleResult();
sampleResult.setSampleLabel("insert");
//请求开始
sampleResult.sampleStart();
try {
// 4、对SQL语句中的问号进行替换
String boyNum = javaSamplerContext.getParameter("boyNum");
String matchNum = javaSamplerContext.getParameter("matchNum");
// int age = javaSamplerContext.getIntParameter("age");
ps.setString(1,boyNum);
ps.setString(2,matchNum);
// 5.执行SQL
int rows = ps.executeUpdate();
if (rows == 1){
//请求成功
sampleResult.setSuccessful(true);
}else{
//请求失败
sampleResult.setSuccessful(false);
}
String response = "插入的行数为:"+rows;
sampleResult.setResponseData(response,"utf-8");
System.out.println("插入的数据行数为:" + rows);
} catch (SQLException e) {
e.printStackTrace();
}
//请求结束
sampleResult.sampleEnd();
return sampleResult;
}
@Override
public void teardownTest(JavaSamplerContext javaSamplerContext) {
// 6.关闭数据库链接
try {
conn.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
/*
* 自定义的参数,它可以获取到Jmeter面板上的参数
*/
@Override
public Arguments getDefaultParameters() {
Arguments arguments = new Arguments();
arguments.addArgument("dbUrl","jdbc:mysql://localhost:3306/test?useUnicode=true&characterEncoding=utf8");
arguments.addArgument("dbUserName","root");
arguments.addArgument("dbPassWord","root");
arguments.addArgument("boyNum","boy-007");
arguments.addArgument("matchNum","111");
return arguments;
}
}
3.1服务
package test;
import java.io.File;
import java.io.PrintWriter;
/**
* Created by April_Chou on 2018/4/10.
*/
public class OutputService {undefined
public static void output(String filename,int a, int b)throws Exception {undefined
PrintWriter out =new PrintWriter(new File(filename));
out.write(a+":"+b);
out.close();
}
}
3.2测试类:
package test;
import org.apache.jmeter.config.Arguments;
import org.apache.jmeter.protocol.java.sampler.JavaSamplerClient;
import org.apache.jmeter.protocol.java.sampler.JavaSamplerContext;
import org.apache.jmeter.samplers.SampleResult;
/**
* Created by April_Chou on 2018/4/10.
*/
public class PerformenceTestimplements JavaSamplerClient {undefined
private SampleResultresults;
private Stringa;
private Stringb;
private Stringfilename;
// 设置传入的参数,可以设置多个,已设置的参数会显示到Jmeter的参数列表中
public ArgumentsgetDefaultParameters() {undefined
Arguments params =new Arguments();
params.addArgument("filename", "0");//设置参数,并赋予默认值0
params.addArgument("a", "0");//设置参数,并赋予默认值0
params.addArgument("b", "0");//设置参数,并赋予默认值0
return params;
}
// 初始化方法,实际运行时每个线程仅执行一次,在测试方法运行前执行
public void setupTest(JavaSamplerContext arg0) {undefined
results =new SampleResult();
}
// 测试执行的循环体,根据线程数和循环次数的不同可执行多次
@Override
public SampleResultrunTest(JavaSamplerContext arg0) {undefined
b = arg0.getParameter("b"); // 获取在Jmeter中设置的参数值
a = arg0.getParameter("a"); // 获取在Jmeter中设置的参数值
filename = arg0.getParameter("filename"); // 获取在Jmeter中设置的参数值
results.sampleStart();// jmeter 开始统计响应时间标记
try {undefined
OutputService test =new OutputService();
test.output(filename,Integer.parseInt(a), Integer.parseInt(b));
results.setSuccessful(true);
// 被测对象调用
}catch (Throwable e) {undefined
results.setSuccessful(false);
e.printStackTrace();
}finally {undefined
results.sampleEnd();// jmeter 结束统计响应时间标记
}
return results;
}
// 结束方法,实际运行时每个线程仅执行一次,在测试方法运行结束后执行
public void teardownTest(JavaSamplerContext arg0) {undefined
}
public static void main(String[] args) {undefined
// TODO Auto-generated method stub
Arguments params =new Arguments();
params.addArgument("a", "0");//设置参数,并赋予默认值0
params.addArgument("b", "0");//设置参数,并赋予默认值0
JavaSamplerContext arg0 =new JavaSamplerContext(params);
PerformenceTest test =new PerformenceTest();
test.setupTest(arg0);
test.runTest(arg0);
test.teardownTest(arg0);
}
}
4.Export为Runnable Jar File
打开IDEA的file -> Project Structure或者快捷方式,进入项目配置页面。如下图:
Artifacts
点击Artifacts,进入Artifacts配置页面,点击 + ,选择如下图的选项
说明:
第一步选择Main函数执行的类,可以进行自动搜索。
第二步选择如图的选项,目的是对第三方Jar包打包时做额外的配置,如果不做额外的配置可不选这个选项(但不保证打包成功)
第三步需要在测试类的目录下,新建一个resources目录,将MANIFEST.MF文件保存在这里面,因为如果用默认缺省值的话,在IDEA12版本下会有bug。
点击OK之后,出现如下图界面,右键点击output root,点击Create Directory,创建一个libs,将所有的第三方JAR放进libs目录下,成功之后,如下图所示:
点击Build->Build Artifacts,选择build
build
就会生成我们需要的jar包。其位置在项目目录的out目录下
jar包成功
5.Jmeter导入
将此jar包放入JMETER_HOME\lib\ext目录
6.打开Jmeter
建议以管理员身份打开JMeter
7.配置Jmeter
创建线程组、Java请求、图形结果、查看结果树、聚合报告、用表格察看结果,进行测试
注意:
这样的Java请求值需要你在相应的盘创建一个文件夹,否则会运行失败,我这里是D盘创建了一个test的文件夹。
界面
Java请求里的参数可以使用函数助手进行填充,如下图:
线程组设置
8.运行结果