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目录下两个两个子目录

java性能测试登录测试 java性能测试脚本_java性能测试登录测试

 

 2.2IDEA操作

1、打开 File -> Project Structure (Ctrl + Shift + Alt + S)或者使用快捷键

 

java性能测试登录测试 java性能测试脚本_java性能测试登录测试_02

java性能测试登录测试 java性能测试脚本_sql_03

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配置页面,点击 + ,选择如下图的选项

java性能测试登录测试 java性能测试脚本_sql_04

 

 

 

java性能测试登录测试 java性能测试脚本_jar包_05

说明:

第一步选择Main函数执行的类,可以进行自动搜索。

第二步选择如图的选项,目的是对第三方Jar包打包时做额外的配置,如果不做额外的配置可不选这个选项(但不保证打包成功)

第三步需要在测试类的目录下,新建一个resources目录,将MANIFEST.MF文件保存在这里面,因为如果用默认缺省值的话,在IDEA12版本下会有bug。

点击OK之后,出现如下图界面,右键点击output root,点击Create Directory,创建一个libs,将所有的第三方JAR放进libs目录下,成功之后,如下图所示:

 

 

点击Build->Build Artifacts,选择build

build

java性能测试登录测试 java性能测试脚本_java性能测试登录测试_06

 

 就会生成我们需要的jar包。其位置在项目目录的out目录下

java性能测试登录测试 java性能测试脚本_sql_07

 

 

 jar包成功

5.Jmeter导入

将此jar包放入JMETER_HOME\lib\ext目录

6.打开Jmeter

建议以管理员身份打开JMeter

7.配置Jmeter

创建线程组、Java请求、图形结果、查看结果树、聚合报告、用表格察看结果,进行测试

注意:

这样的Java请求值需要你在相应的盘创建一个文件夹,否则会运行失败,我这里是D盘创建了一个test的文件夹。

界面

java性能测试登录测试 java性能测试脚本_jar包_08

Java请求里的参数可以使用函数助手进行填充,如下图:

 

java性能测试登录测试 java性能测试脚本_java性能测试登录测试_09

线程组设置

java性能测试登录测试 java性能测试脚本_java_10

 

 

8.运行结果

 

java性能测试登录测试 java性能测试脚本_jar包_11