zoukankan      html  css  js  c++  java
  • jmeter中beanshell postprocessor结合fastjson库提取不确定个数的json参数

    在项目实践中,遇到了这样一个问题。用jmeter作http接口测试,需要的接口参数个数是不确定的。也就是说,在每次测试中,根据情况不同,可能页面中的列表中所含的参数个数是不确定的,那么要提取的参数个数也是不确定的,可能是1个,也可能是2个或3个,甚至更多。

    例如,返回的接口消息json消息可能如下

    {
    "data": {
    "records": [{
    "DEVICE_TYPE": 194,
    "DEVICE_STATUS": 0,
    "ORG_NAME": "**省",
    "LONGITUDE": "0.000000",
    "DEVICE_NAME": "测试1",
    "DEVICE_ADDR": "",
    "DEVICE_ID_INT": "13",
    "HAVE_VIDEO": "true",
    "OWNER": "-1",
    "ORIGINAL_DEVICE_ID": "44000000011325******",
    "PLATFORM_ID": "44000000052005******",
    "ORG_CODE": "44",
    "TYPE": "1",
    "DEVICE_ID": "44000000011325******",
    "LATITUDE": "0.000000"
    }, {
    "DEVICE_TYPE": 194,
    "DEVICE_STATUS": 0,
    "LONGITUDE": "0.000000",
    "DEVICE_NAME": "测试2",
    "DEVICE_ADDR": "",
    "DEVICE_ID_INT": "10",
    "HAVE_VIDEO": "true",
    "OWNER": "1",
    "ORIGINAL_DEVICE_ID": "44010000001320******",
    "PLATFORM_ID": "44000000052005******",
    "ORG_CODE": "4401",
    "TYPE": "1",
    "DEVICE_ID": "440100000013200******",
    "LATITUDE": "0.000000"
    }, {
    "DEVICE_TYPE": 194,
    "DEVICE_STATUS": 0,
    "LONGITUDE": "0.000000",
    "DEVICE_NAME": "测试3",
    "DEVICE_ADDR": "",
    "DEVICE_ID_INT": "11",
    "HAVE_VIDEO": "true",
    "OWNER": "1",
    "ORIGINAL_DEVICE_ID": "44010000001320******",
    "PLATFORM_ID": "44000000052005******",
    "ORG_CODE": "4401",
    "TYPE": "1",
    "DEVICE_ID": "44010000001320******",
    "LATITUDE": "0.000000"
    }],
    "count": 3,
    "usetime": 0
    }
    }

    说是可能,是因为存在多种不同的返回结果,上述json消息中返回的有3个设备信息,但也可能是1个,也可能是2个,或者是更多,具体的数值要根据页面状况。我们的任务是从返回的json消息中提取所有的设备ID号(DEVICE_ID)。

    个人认为,这种初始环境的不确定性本来是自动化测试的大忌,会给脚本的成功运行带来很大的风险,但在实际应用中,由于测试环境并非是个人独占的,每次测试时,页面配置的变化是可以理解的存在。为了提高脚本的健壮性,最好的办法是每次根据实际页面的返回值,提取其中的所有参数,然后拼接好,存jmeter的内部变量以供后续接口使用。要做好这一点,光用现成的jmeter工具,尤其是我们常用的json提取器(json extractor)肯定是无法做到了,至少做不到参数拼接后续处理。如果针对不同的场景,用if控制器来做区分,又会显得太累赘。

    作为一个世界级的以java为基础的开源工具,肯定有其他的办法,个人感觉,jmeter搭载了beanshell就好像robot framework搭载了evaluate,一下子世界宽广了。作为一个半吊子,jmeter还没搞透彻,java也没用过,又要开始研究beanshell也是有点拼。只好求助万能的度娘。网上提到了一种beanshell postprocessor结合fastjson库的方法,可以做json的快速高效提取。

    先把配置环境说一下,我用到的是jmeter5.0,fastjson用到的是1.2.47的版本,下载地址如下http://repo1.maven.org/maven2/com/alibaba/fastjson/1.2.47/,需要把下载下来的jar包放到jmeter的安装路径的lib/ext库中,并且在jmeter test plan中引用这个jar包

    在需要提取的页面添加BeanShell后置处理器

    script编写如下:

    import com.alibaba.fastjson.JSON;
    import com.alibaba.fastjson.JSONArray; 
    import com.alibaba.fastjson.JSONObject;
    // 获取数据
    //log.info("---something---");
    String response = prev.getResponseDataAsString();  // 获取Response
    //log.info("Respnse is " + response);
    JSONObject responseObj = JSON.parseObject(response);  // 整个Response作为JSON对象
    JSONObject resObj = responseObj.getJSONObject("data");  // 获取某一部分对象。即,json串中{}的内容
    //log.info("resObj is " + resObj);
    JSONArray listArray = resObj.getJSONArray("records");  // 获取列表。即,json串中[]的内容
    // 保存数据
    // 1) 列表
    //log.info("listArray is "+ listArray);
    //log.info("array[0] is "+ listArray.getJSONObject(0).getString("DEVICE_ID"));
    int len = listArray.size();
    String[] temp = new String[len];
    StringBuffer sBuffer = new StringBuffer("");
    for(int i=0;i<len;i++){
        temp[i]= listArray.getJSONObject(i).getString("DEVICE_ID");
        sBuffer.append(temp[i]).append(",");
    //    log.info("sbuffer---"+sBuffer);
        //vars.put("deviceId"+i, temp[i]);// 保存到JMeter变量中
        //log.info("deviceIdAll===",deviceIdAll);
    }
    String keywordStr = sBuffer.deleteCharAt(sBuffer.length() - 1).toString();
    vars.put("deviceId",keywordStr);

    主要用到的API释义如下:

    1. 将json字符串反序列化成JSON对象

    JSONObject com.alibaba.fastjson.JSON.parseObject(String text)

    2.根据key 得到json中的json数组

    JSONArray com.alibaba.fastjson.JSONObject.getJSONArray(String key)

    3. 根据下标拿到json数组的json对象

    JSONObject com.alibaba.fastjson.JSONArray.getJSONObject(int index)

    4.. 根据key拿到json的字符串值

    String com.alibaba.fastjson.JSONObject.getString(String key)

    更详细的fastjson的详细API参考了这篇文章:https://www.cnblogs.com/qiaoyeye/p/7730288.html

    总结一下,此次踩过的坑主要有如下:

    1.字符串是定义好之后就不能变更的,属于常量,和之前使用python的经验不同,在java中对于字符串,是不能使用拼接操作的,要不断写字符串,得用StringBuffer类。网上找到了这段话:

    当对字符串进行修改的时候,需要使用 StringBuffer 和 StringBuilder 类。

    和 String 类不同的是,StringBuffer 和 StringBuilder 类的对象能够被多次的修改,并且不产生新的未使用对象。

    StringBuilder 类在 Java 5 中被提出,它和 StringBuffer 之间的最大不同在于 StringBuilder 的方法不是线程安全的(不能同步访问)。

    由于 StringBuilder 相较于 StringBuffer 有速度优势,所以多数情况下建议使用 StringBuilder 类。然而在应用程序要求线程安全的情况下,则必须使用 StringBuffer 类。

    2.运行时在控制台总是提示有这样的错误"Jmeter ERROR o.a.j.u.BeanShellInterpreter: Error invoking bsh method",我一直在怀疑是fastjson这个第三方库的使用有问题,最后发现是由于脚本中存在代码错误导致的,至于错误的原因,自己一行一行地找,所以在脚本中我加了无数的log.info,不得不说,beanshell脚本缺乏IDLE的编码环境,确实很不友好。

    总之,通过这次的实践,进一步扩充了自己的知识面,再一次体会到,边走边看,坚持往前走的魅力所在。不要想着准备好所有的必要条件再出发,要用事情来推动自己去创造条件,任何经历都是成长。工作与生活皆如此!

     
     
  • 相关阅读:
    VSCode中快捷键位设置优化
    错误检测(4) CRC
    linux 中的col命令
    使Chrome的字体渲染更清晰
    并发工具类(三)控制并发线程的数量 Semphore
    并发工具类(二)同步屏障CyclicBarrier
    并发工具类(一)等待多线程的CountDownLatch
    显式锁(四)Lock的等待通知机制Condition
    显式锁(三)读写锁ReadWriteLock
    显式锁(二)Lock接口与显示锁介绍
  • 原文地址:https://www.cnblogs.com/jingmu/p/10923254.html
Copyright © 2011-2022 走看看