Jmeter二次开发——自定义函数
在之前的博文中,Jmeter二次开发——基于Java请求,已介绍了Jmeter二次开发的基础情况,上次分享的是java请求开发,今天来分享下Jmeter中的函数开发。聊到Jmeter的函数,知道Jmeter使用的博友肯定很熟悉。Jmeter自带一个函数库,有很多的函数,比如:__P,__Random,函数助手给我们提供了很多的方便之处。函数助手使用如下所示:

但有些时候,自带的函数满足不了真实的测试场景,比如:生成随机手机号。常规做法,应该是设定手机号区号的固定值,再通过__Random函数生成8位随机数,从而拼接成一个手机号,这样的做法的确可以满足,但要想手机号的区段也是随机的呢,是不是就不太好处理了。那就用函数二次开发试试。
函数二次开发
创建以functions结尾的包
这个是特别需要注意点,以.functions结尾,正常创建包即可。
类继承AbstractFunction
二次开发时,新建的类,需要继承AbstractFunction,这个也是需要注意的。至于为什么需要继承AbstractFunction,看源码就能明白,源码如下所示:
public abstract class AbstractFunction implements Function {  
    public AbstractFunction() {  
    }  
    public abstract String execute(SampleResult var1, Sampler var2) throws InvalidVariableException;
    public String execute() throws InvalidVariableException {
        JMeterContext context = JMeterContextService.getContext();
        SampleResult previousResult = context.getPreviousResult();
        Sampler currentSampler = context.getCurrentSampler();
        return this.execute(previousResult, currentSampler);
    }
    public abstract void setParameters(Collection<CompoundVariable> var1) throws InvalidVariableException;
    public abstract String getReferenceKey();
    protected JMeterVariables getVariables() {
        return JMeterContextService.getContext().getVariables();
    }
    protected void checkParameterCount(Collection<CompoundVariable> parameters, int min, int max) throws InvalidVariableException {
        int num = parameters.size();
        if (num > max || num < min) {
            throw new InvalidVariableException(this.getReferenceKey() + " called with wrong number of parameters. Actual: " + num + (min == max ? ". Expected: " + min + "." : ". Expected: >= " + min + " and <= " + max));
        }
    }
    protected void checkParameterCount(Collection<CompoundVariable> parameters, int count) throws InvalidVariableException {
        int num = parameters.size();
        if (num != count) {
            throw new InvalidVariableException(this.getReferenceKey() + " called with wrong number of parameters. Actual: " + num + ". Expected: " + count + ".");
        }
    }
    protected void checkMinParameterCount(Collection<CompoundVariable> parameters, int minimum) throws InvalidVariableException {
        int num = parameters.size();
        if (num < minimum) {
            throw new InvalidVariableException(this.getReferenceKey() + " called with wrong number of parameters. Actual: " + num + ". Expected at least: " + minimum + ".");
        }
    }
    protected final void addVariableValue(String value, CompoundVariable[] values, int index) {
        if (values.length > index) {
            String variableName = values[index].execute().trim();
            if (StringUtils.isNotEmpty(variableName)) {
                JMeterVariables vars = this.getVariables();
                if (vars != null) {
                    vars.put(variableName, value);
                }
            }
        }
    }
}
参数解释
getArgumentDesc()
获取界面所要显示的参数说明
execute()
函数的主体业务
getReferenceKey()
获取函数的名称
setParameters()
设置参数,接收用户传递的参数
checkParameterCount()
检测参数数量是否准确
函数开发
获取函数的名称
名称自定义,如下所示:
private static final String key = "__XXX";
这里需要注意的是:函数开头是以2个下划线开头。
名称定义好了,那如何获取呢?就用我们刚才说的方法获取即可,如下所示:
@Override
public String getReferenceKey() {
    return key;
}
获取界面所要显示的参数说明
在Jmeter的函数助手中,对应函数都有对应的参数说明,如下所示:

那如何配置能实现呢?代码如下:
private final static List<String> args = new LinkedList<String>();
static{
      args.add("界面参数");
}
如果有多个参数怎么办?多个参数,多个args.add即可
获取参数名称,同样用刚才介绍的方法获取即可,如下所示:
@Override
public List<String> getArgumentDesc() {
    return args;
}
获取参数值
@Override
    public void setParameters(Collection<CompoundVariable> args0) throws InvalidVariableException {
      //检测用户调用函数时,检查参数个数,个数不对则报错
      checkParameterCount(args0,3);
      Object[] params = args0.toArray();
      //转换只为string
      telNum = ((CompoundVariable)params[0]).execute();
      start = ((CompoundVariable)params[1]).execute();
      end = ((CompoundVariable)params[2]).execute();
  }
获取参数值中,可以检测函数的入参个数是否准确,不准确则会报错,报错信息如下所示:

函数的主体业务
介绍到这,就是函数的核心内容了,该函数要实现什么功能,就是在该方法中处理,示例代码如下所示:
    @Override
    public String execute(SampleResult sampleResult, Sampler sampler) throws InvalidVariableException {
        SampleResult sampleResult1 = new SampleResult();
        try {
            sampleResult1.sampleStart();
            int index=getNum(0,telFirst.length-1);
            String telNum = telFirst[index];
            String two = String.valueOf(getNum(1, 888) + 10000).substring(1);
            String three = String.valueOf(getNum(1, 9100) + 10000).substring(1);
            tel = telNum + two + three;
            logger.info("手机号区段:"+ telNum +" 随机生成的手机号是:" + tel);
            if (varName != null) {
                JMeterVariables vars = getVariables();
                final String varTrim = varName.execute().trim();
                if (vars != null && varTrim.length() > 0) {
                    vars.put(varTrim, telNum);
                }
            }
            sampleResult1.setResponseData("手机号区段:"+ telNum +" 随机生成的手机号是:" + tel,"utf-8");
            sampleResult1.setSuccessful(true);
        }catch (Exception e){
            sampleResult.setSuccessful(false);
            e.printStackTrace();
        }finally {
            sampleResult1.sampleEnd();
        }
        return tel;
    }
代码调试
写到这里,基本完成了,但还是得测试下,功能是否正常,如果先打jar包,丢到Jmeter中,发现有bug的话,来来回回处理,就折腾了,所以还是需要先测试下的。
在test下新建测试类,示例代码如下所示:
import org.junit.Test;
public class Function_Test {
    @Test
    public void phoneTest() throws Exception {
        RandomPhoneJmeterFunctions randomPhone= new RandomPhoneJmeterFunctions();
        String phoneString = randomPhone.execute();
        System.out.println("随机手机号:" + phoneString);
    }
}
测试代码很简单,运行测试类,没有报错并打印出手机号,则说明没有问题。运行后的结果如下所示:

生成jar包
生成jar包就不重复讲了,可以看以前的博文,IDEA的基本操作——导入导出jar包
Jmeter中使用
代码写好后,自然是要在jmeter中验证下功能的,我们将生成的jar包放到jmeter的\lib\ext文件夹下,如果jmeter已启用,则需要重启哦,不然不会生效。
打开jmeter后,使用函数助手,看新开发的函数是否有展示,如下所示:

生成函数变量,操作如下所示:

新建线程组,并添加http请求,验证码生成的手机号是不是随机的,运行后,查看结果树,如下所示:

也可以通过日志查看,开发的时候,加了响应日志,如下所示:

到此,就说明功能没问题了。函数开发按上述步骤就可以完成,遇到不满足测试场景的时候,就可以自己diy一个了。
最后附上完整代码,如下所示:
private static Logger logger = LogManager.getLogger(RandomPhoneJmeterFunctions.class.getName());
    private String tel;
    //定义函数名称
    private static final String KEY = "__RandomPhone";
    //定义函数界面显示的参数名称
    private static final List<String> desc = new LinkedList<String>();
    static{
        desc.add("界面参数");
    }
    private static final String[] telFirst = "134,135,136,137,138,139,150,151,152,157,158,159,130,131,132,155,156,133,153 ".split(",");
    private CompoundVariable varName;
    //业务主逻辑
    @Override
    public String execute(SampleResult sampleResult, Sampler sampler) throws InvalidVariableException {
        SampleResult sampleResult1 = new SampleResult();
        try {
            sampleResult1.sampleStart();
            int index=getNum(0,telFirst.length-1);
            String telNum = telFirst[index];
            String two = String.valueOf(getNum(1, 888) + 10000).substring(1);
            String three = String.valueOf(getNum(1, 9100) + 10000).substring(1);
            tel = telNum + two + three;
            logger.info("手机号区段:"+ telNum +" 随机生成的手机号是:" + tel);
            if (varName != null) {
                JMeterVariables vars = getVariables();
                final String varTrim = varName.execute().trim();
                if (vars != null && varTrim.length() > 0) {
                    vars.put(varTrim, telNum);
                }
            }
            sampleResult1.setResponseData("手机号区段:"+ telNum +" 随机生成的手机号是:" + tel,"utf-8");
            sampleResult1.setSuccessful(true);
        }catch (Exception e){
            sampleResult.setSuccessful(false);
            e.printStackTrace();
        }finally {
            sampleResult1.sampleEnd();
        }
        return tel;
    }
    //获取参数值
    @Override
    public void setParameters(Collection<CompoundVariable> args0) throws InvalidVariableException {
        //检测用户调用函数时,检测参数个数
        checkParameterCount(args0,1);
        Object[] params = args0.toArray();
        if (params.length > 0) {
            varName = (CompoundVariable) params[0];
        } else {
            varName = null;
        }
    }
    //获取函数的名称
    @Override
    public String getReferenceKey() {
        return KEY;
    }
    //获取界面所要显示的参数说明
    @Override
    public List<String> getArgumentDesc() {
        return desc;
    }
    private static int getNum(int start,int end)
    {
        return (int)(Math.random()*(end-1));
    }
Jmeter二次开发——自定义函数的更多相关文章
- JMeter二次开发(2)-编写 JSON Assertion 插件
		本篇文章主要介绍如何对JMeter进行二次开发,添加自己所需的功能.这里以Json验证为例进行说明.在web接口测试过程中,JSON的应用已经非常普遍,但原声的JMeter并没有提供Json及Json ... 
- JMeter二次开发(1)-eclipse环境配置及源码编译
		1.下载src并解压 http://jmeter.apache.org/download_jmeter.cgi 2.获取所需jar包,编译 ant download_jars ant instal ... 
- Jmeter二次开发
		Jmater函数扩展的步骤1. 导入Jmeter源码,或使用maven项目,引入依赖的jar包 2. 继承AbstractFunction,实现自定义Function 3. 继承JMeterTestC ... 
- 性能测试基础---jmeter二次开发
		·Jmeter的二次开发,常见的有以下几种类型: ·扩展.修改Jmeter已有的组件(源代码) ·扩展.修改Jmeter已有的函数. ·完全自主开发一个新的组件(依赖于Jmeter提供的框架). ·扩 ... 
- Jmeter二次开发——基于Java请求
		简述 这近几年,越来越多非http的协议需要进行性能测试,包括不仅限于各类rpc.mq.缓存等.对于这些协议,市面上可能没有现成的工具可以直接使用,这个时候,我们可以自己动手,通过编写相应的JavaS ... 
- jmeter 二次开发
		基于jmeter的java请求的二次开发 常用的方法: ①.addArgument("name", "value") 定义参数 ②.sampleStart ... 
- NX二次开发CreateDialog函数在UI.hxx文件和WinUser.h中的冲突【转载】
		文章出自https://blog.csdn.net/qq_41843732/article/details/91422764 在UG二次开发中,若使用MFC库,一旦加上#include<Afx. ... 
- Jmeter二次开发实现自定义functions函数(九)
		在Jmeter->选项->函数助手对话框中我们可以看到Jmeter内置的一些常用函数,但考虑到测试过程中的实际情况,我们经常需要在脚本引用或者实现自定义的函数.那么如何在"函数助 ... 
- jmeter 二次开发---实现自定义函数插件
		1.前提: 有时候,Jmeter自带的函数,可能不能满足于业务的需求,这时候,我们可以自己写一个函数插件: 2.创建maven工程 一直next,输入GroupID,ArtifactId->fi ... 
随机推荐
- windows 10 1909 无法启用 .NET Framework 解决
			安装某应用,运行提示: 应用程序无法正常启动(0xc0000135) 应该是缺少 .net framework. 控制面板-程序-"启用或关闭windows功能" 勾选.NET F ... 
- uwsgi 的app变量名称必须为application
			from myproject import app as application if __name__ == "__main__": application.run() 否则会提 ... 
- rbd锁引起kvm虚拟机无法启动的故障
			前言 环境因为一些问题(网络,或者磁盘,或者其它各种异常),引起了集群的状态的一些变化,变化之后,集群的某些虚拟机正常某些虚拟机出现异常,异常现象就是无法启动 特别是win server2008 ,会 ... 
- 找和为K的两个元素
			总时间限制:1000ms 内存限制: 65536kB 描述 在一个长度为n(n < 1000)的整数序列中,判断是否存在某两个元素之和为k. 输入 第一行输入序列的长度n和k,用空格分开. 第二 ... 
- 【Idea插件】kotlin的orm框架一键生成代码框架
			@font-face { font-family: octicons-link; src: url("data:font/woff;charset=utf-8;base64,d09GRgAB ... 
- 关于新创公司所需的icp,网文,软著和备案的申请
			刚从一个集团离职来到了创业团队,前期是什么都没有,甚至是公司名字都不知道,哈哈.所以就有了后面的坑踩了一遍又一遍.刚开始是在霍尔果斯注册,结果办icp费了半年的时间,东找西找还没下证.又碰上新疆严查不 ... 
- asp.net webapi关闭https配置
			将s去掉就行 
- MVC中使用T4模板
			参考博文 http://www.cnblogs.com/heyuquan/archive/2012/07/26/2610959.html 图片释义 1.简单示例,对基本的模块标记 2.根据上图生成的类 ... 
- easyui中连接按钮样式
			方法1. <a href="otherpage.php" class="easyui-linkbutton" data-options="ico ... 
- 多线程,线程类三种方式,线程调度,线程同步,死锁,线程间的通信,阻塞队列,wait和sleep区别?
			重难点梳理 知识点梳理 学习目标 1.能够知道什么是进程什么是线程(进程和线程的概述,多进程和多线程的意义) 2.能够掌握线程常见API的使用 3.能够理解什么是线程安全问题 4.能够知道什么是锁 5 ... 
