博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
jmeter内对response值的处理
阅读量:5877 次
发布时间:2019-06-19

本文共 10452 字,大约阅读时间需要 34 分钟。

jmeter内对response值的处理

1. 正则表达式提取器

在取样器上右键,选择正则表达式提取器,如图:

img_ddfe20d2fb9b7019f2b512aef0869a4f.jpe
image

接下来看一下正则表达式提取器的界面:

img_92e0c56e75dc42bc2f3b5f2e656a3533.jpe
image

在这里对页面各字段进行一下说明,从上往下:

1. Apply to,是指你提取的变量要应用的范围,需要注意的是,Jmeter Variable是指应用到全局,也就是跨线程的。

2. 要检查的响应字段,是指你的正则表达式的提取范围,主体应该是整个response,单选body就是指在body里找,其它选项根据字面意思,就不再赘述。

3. 引用名称,是我们自定义的一个变量名称,也就是提取出来的值的变量名称,例如我给它定义为“var”,那么正则提取出来的值就存在var里,之后如果要用,就使用el表达式,也就是${var(刚刚定义的名称)}的方式来使用。

4. 正则表达式,这里写提取变量用的。

5. 模板,这里是指你提取多少个变量(如果在正则表达式中有多个提取表达式),一般我们提取一个就够了,模板采用$N$的方式,N代表数量,例如,我们设立一个只提取1个值的模板,就写作$1$。

6. 匹配数字,也就是提取匹配到的第几个值,例如写1就是从匹配到的值里选取第一个。

7. 缺省值,也就是如果提取不到,那么默认给一个值替换提取结果。

在这里举一个例子如下:

现在有一段response值,

{"errorCode":"","success":true,"message":"","data":{"scoreId":"98","templateId":"","title":"新建权重评分表-auto-test","type":"1","state":"2","containers":[{"id":"692","text":"维度1","components":[{"id":"2196","text":"第一个指标","percent":"25","type":"1","options":[{"id":"6978","text":"好","score":"100","sort_num":1},{"id":"6979","text":"较好","score":"80","sort_num":2},{"id":"6980","text":"一般","score":"60","sort_num":3},{"id":"6981","text":"较差","score":"40","sort_num":4},{"id":"6982","text":"差","score":"0","sort_num":5}]},{"id":"2197","text":"第二个指标","percent":"25","type":"2","options":[{"id":"6983","text":"评分标准1","score":"100","sort_num":1}]}]},{"id":"693","text":"维度2","components":[{"id":"2198","text":"第三个指标","percent":"25","type":"2","options":[{"id":"6984","text":"评分标准2","score":"100","sort_num":1}]},{"id":"2199","text":"第四个指标","percent":"25","type":"1","options":[{"id":"6985","text":"好","score":"100","sort_num":1},{"id":"6986","text":"较好","score":"80","sort_num":2},{"id":"6987","text":"一般","score":"60","sort_num":3},{"id":"6988","text":"较差","score":"40","sort_num":4},{"id":"6989","text":"差","score":"0","sort_num":5}]}]}]}}

我们需要从中把维度1对应的id提取出来,那么可以按照如下参数(."id":"(.+?)","text":"维度1"*)提取:

img_1661f7871c46d0bb16ed335df632ab97.jpe
image

附上正则表达式的图:

img_052a2cb5dd42669cbb6e03b51bb8a143.png
image

2. 使用beanshell提取

在取样器上右键,选择beanshell postprocessor,如图:

img_4277b39b3f6dfd8d96bf1fce86b30426.jpe
image

接下来看一下beanshell的界面:

img_78b3a134bb372f86454688e952823e89.jpe
image

beanshell就是一个运行Java的微环境(大概可以这么理解),在Script框里我们写java代码,jmeter执行到这里的时候就会执行里面的java代码,beanshell里面会有一些jmeter内置的变量,用于jmeter和你写的Java代码交互,常用的大约有这么几个:

1. log:写入信息到jmeber.log文件,使用方法:log. info(“This is log info!”);

2. vars - (JMeterVariables):操作jmeter变量,这个变量实际引用了JMeter线程中的局部变量容器(本质上是Map),它是测试用例与BeanShell交互的桥梁,常用方法:

  • vars.get(String key):从jmeter中获得变量值
  • vars.put(String key,String value):数据存到jmeter变量中

3. prev - (SampleResult):获取前面的sample返回的信息,常用方法:

  • getResponseDataAsString():获取响应信息
  • getResponseCode() :获取响应code

在这里根据我在工作里遇到的一个实战来说明

首先说明下需求,现在存在一个接口是post类型,http协议的,他需要若干个参数用以传参,其中一个参数“scoreData”是json类型的数据,数据结构如下:

{

"645(题目的id)":{"id":"2546(选项的id)","percent":"25(分值,固定值)"},
"646(同上)":{"id":"2551(同上)","percent":"24.75(同上)","value(该参数只有在题目的type=2的时候才会出现)":"99(分值,固定值)"}
}

说明一下这个数据结构,我们的问卷可能会存在多个题目,上面case里的645,646就代表是两道题目,题目的类型分为选择题和填分题,选择题的type是1,我们只取第一个选项的id(也就是说传参代表勾选了第一项);填分题的type是2,我们这里固定填99分,也就是value是99;percent这个字段是根据权重计算出来的值,这里不多做说明,就当做它的值也是一个固定的值。

数据全部来源于前一个接口的返回值,返回值如下:

{"success":true,"message":"","errorCode":"","data":[{"submit_state":"0","score_id":"f5","travel_id":"5","travel_title":"auto-test-zyj1502356486388","inspect_time":"2033-08-23","category_codes":"2266","suggest_content":"","my_score":"0","categoryName":"商业包装设计与施工","scoreData":{"fbScoreId":"f5","templateId":"","title":"新建权重评分表-auto-test-zyj","type":1,"containers":[{"id":"8","text":"维度1","components":[{"id":"14","text":"第一个指标","percent":"25","type":"1","options":[{"id":"38","text":"好","score":"100","sort_num":1},{"id":"39","text":"较好","score":"80","sort_num":2},{"id":"40","text":"一般","score":"60","sort_num":3},{"id":"41","text":"较差","score":"40","sort_num":4},{"id":"42","text":"差","score":"0","sort_num":5}],"select":""},{"id":"15","text":"第二个指标","percent":"25","type":"2","options":[{"id":"43","text":"评分标准1","score":"100","sort_num":1}],"select":""}]},{"id":"9","text":"维度2","components":[{"id":"16","text":"第三个指标","percent":"25","type":"2","options":[{"id":"44","text":"评分标准2","score":"100","sort_num":1}],"select":""},{"id":"17","text":"第四个指标","percent":"25","type":"1","options":[{"id":"45","text":"好","score":"100","sort_num":1},{"id":"46","text":"较好","score":"80","sort_num":2},{"id":"47","text":"一般","score":"60","sort_num":3},{"id":"48","text":"较差","score":"40","sort_num":4},{"id":"49","text":"差","score":"0","sort_num":5}],"select":""}]}]},"travelDetail":{"travel_id":"5","score_id":"f5","travel_title":"auto-test-zyj1502356486388","company_name":"上海东方雨虹防水技术有限责任公司","supplier_type":"2","tax_qualify":"2","organizer_id":"195104","inspect_time":"2033-08-23","scoreName":"新建权重评分表-auto-test-zyj","supplierTypeName":"代理商","taxQualifyName":"小规模纳税人","organizerName":"125","Teams":"zyj账号01、zyj账号02","Locations":[{"name":"办公室","remark":"备注1"},{"name":"地点一","remark":"备注2"}],"QualifyFiles":[{"name":"营业执照","remark":"备注3"},{"name":"地点二","remark":"备注4"}]},"travelPhoto":[],"photoDescription":{"1":[{"address":"地点一"},{"address":"办公室"}],"2":[{"address":"地点二"},{"address":"营业执照"}]},"photo_count":0,"score_item_total_count":"4"},{"submit_state":"0","score_id":"f4","travel_id":"4","travel_title":"auto-test-zyj1502351094326","inspect_time":"2033-08-23","category_codes":"2266","suggest_content":"","my_score":"0","categoryName":"商业包装设计与施工","scoreData":{"fbScoreId":"f4","templateId":"","title":"新建权重评分表-auto-test-zyj","type":1,"containers":[{"id":"6","text":"维度1","components":[{"id":"10","text":"第一个指标","percent":"25","type":"1","options":[{"id":"26","text":"好","score":"100","sort_num":1},{"id":"27","text":"较好","score":"80","sort_num":2},{"id":"28","text":"一般","score":"60","sort_num":3},{"id":"29","text":"较差","score":"40","sort_num":4},{"id":"30","text":"差","score":"0","sort_num":5}],"select":""},{"id":"11","text":"第二个指标","percent":"25","type":"2","options":[{"id":"31","text":"评分标准1","score":"100","sort_num":1}],"select":""}]},{"id":"7","text":"维度2","components":[{"id":"12","text":"第三个指标","percent":"25","type":"2","options":[{"id":"32","text":"评分标准2","score":"100","sort_num":1}],"select":""},{"id":"13","text":"第四个指标","percent":"25","type":"1","options":[{"id":"33","text":"好","score":"100","sort_num":1},{"id":"34","text":"较好","score":"80","sort_num":2},{"id":"35","text":"一般","score":"60","sort_num":3},{"id":"36","text":"较差","score":"40","sort_num":4},{"id":"37","text":"差","score":"0","sort_num":5}],"select":""}]}]},"travelDetail":{"travel_id":"4","score_id":"f4","travel_title":"auto-test-zyj1502351094326","company_name":"上海东方雨虹防水技术有限责任公司","supplier_type":"2","tax_qualify":"2","organizer_id":"195104","inspect_time":"2033-08-23","scoreName":"新建权重评分表-auto-test-zyj","supplierTypeName":"代理商","taxQualifyName":"小规模纳税人","organizerName":"125","Teams":"zyj账号01、zyj账号02","Locations":[{"name":"办公室","remark":"备注1"},{"name":"地点一","remark":"备注2"}],"QualifyFiles":[{"name":"营业执照","remark":"备注3"},{"name":"地点二","remark":"备注4"}]},"travelPhoto":[],"photoDescription":{"1":[{"address":"地点一"},{"address":"办公室"}],"2":[{"address":"地点二"},{"address":"营业执照"}]},"photo_count":0,"score_item_total_count":"4"}]}

要提取数据,尤其是这么大的json数据,首先我们要把这个json的数据结构分析清楚,我推荐大家把这个json放到里解析一下结构,这样会更清楚,例如bejson,解析出来如下(部分要点截图):

img_3c7ff3b103debb54e45dd2caf3d16cab.jpe
image

我们现在需要做两件事:

  1. 取值,具体是取每个components下面id的值(也就是题目的id),以及每个options下面第一个id的值(也就是选项的id);
  2. 将取到的值组合成需要的数据结构再输出。

对于json的处理,我们需要用到额外的包,里有,在java选项下面,第一个JSON-java就是,这个包叫做org.json,我们下来之后,把这个包放到jmeter的apache-jmeter\lib\ext下面,这样就可以用了。

简单介绍下org.json,这部分建议看一下;

JSONObject

是一个无序的键/值对集合。

  • 它的表现形式是一个包裹在花括号的字符串,键和值之间使用冒号隔开,键值和键值之间使用逗号隔开。
  • 内在形式是一个使用get()和opt()方法通过键来访问值,和使用put()方法通过键来添加或者替代值的对象。
  • 值可以是任何这些类型:Boolean,JSONArray,JSONObject,Number和String,或者JOSONObject.NULL对象。

JSONArray

是一个有序的序列值。

  • 它的表现形式是一个包裹在方括号的字符串,值和值之间使用逗号隔开。
  • 内在形式是一个使用get()和opt()方法通过索引来访问值,和使用put()方法来添加或修改值的对象。
  • 值可以是任何这些类型:Boolean,JSONArray,JSONObject,Number,和String,或者JSONObject.NULL对象。

我们大约只用到这两种就可以了

这里直接放出处理的源码:

import java.util.HashMap;import java.util.Map;import org.json.*;        System.out.println("test................................................................");     //提取返回值作为变量response_data    String response_data = prev.getResponseDataAsString();    //解析json    JSONObject data_obj = new JSONObject(response_data);            JSONArray jsonarray = data_obj.getJSONArray("data");    JSONObject scoreData = null;    int id = Integer.parseInt(vars.get("travel_id"));    System.out.println("id="+id);       for (int i = 0; i < jsonarray.length(); i++) {         JSONObject jsonobj = jsonarray.getJSONObject(i);        System.out.println(jsonarray.optString(i));        System.out.println(jsonobj.getInt("travel_id"));        System.out.println(jsonobj.getString("travel_title"));        int travle_id = jsonobj.getInt("travel_id");       if(travle_id==id){              scoreData = jsonobj.getJSONObject("scoreData");            }                   }       JSONArray containers = scoreData.getJSONArray("containers");    Map data = new HashMap();       for (int i = 0; i < containers.length(); i++){                JSONArray component_ary = containers.getJSONObject(i).getJSONArray("components");                     for (int j = 0; j < component_ary.length(); j++){                System.out.println(component_ary.optString(j));              System.out.println(component_ary.getJSONObject(j).get("id"));              String data_id = component_ary.getJSONObject(j).get("id").toString();             JSONArray options = component_ary.getJSONObject(j).getJSONArray("options");                 String score_id = options.getJSONObject(0).get("id").toString();             System.out.println(options.getJSONObject(0).get("id"));               Map score = new HashMap();              score.put("id", score_id);             score.put("percent", "25");             if(component_ary.getJSONObject(j).getInt("type")==2){                 score.put("value", "99");                                  }             System.out.println(score);              data.put(data_id, score);             System.out.println(data);                             }                    }    //将map类型转换为json字符    JSONObject putdata = new JSONObject(data);    System.out.println(putdata);    //将putdata输出到jmeter变量里    vars.put("putdata", JSONObject.valueToString(putdata));

运行的结果(不是用上面的参数值运行,是我脚本实例的结果,但数据结构是一样的),

putdata的值:putdata={"671":{"id":"2660","percent":"25","value":"99"},"672":{"id":"2661","percent":"25","value":"99"},"673":{"id":"2662","percent":"25"},"670":{"id":"2655","percent":"25"}},这个值被作为变量“putdata”给jmeter使用,我们可以在debugSampler看到这个值:

img_9608261b1b2d18760a4748fb5ce1a394.jpe
image

我们可以在接口里使用这个值了,如图:

img_958141ca4ee7de5b52bc8c8fbe1f3d3f.jpe
image

一些tips

1. 新建处理器的时候,注意选择前置还是后置,前置是指在请求之前进行处理;后置是指在请求之后进行处理。

2. 调试脚本,可以新建一个debugSampler,脚本运行之后,我们可以在查看结果树里查看他的响应数据来观察变量的值;如果是beanshell,可以先在编译器里调试好再复制进去,或者直接打包成jar引用。

引用资料

json教程:

beanshell的使用:

转载地址:http://xakix.baihongyu.com/

你可能感兴趣的文章
Android View.onMeasure方法的理解
查看>>
Node.js 爬虫初探
查看>>
ABP理论学习之仓储
查看>>
NestJS 脑图
查看>>
我的友情链接
查看>>
Html body的滚动条禁止与启用
查看>>
Tengine新增nginx upstream模块的使用
查看>>
多媒体工具Mediainfo
查看>>
1-小程序
查看>>
CentOS图形界面和命令行切换
查看>>
HTML5通信机制与html5地理信息定位(gps)
查看>>
Mind_Manager_2
查看>>
手动升级 Confluence - 规划你的升级
查看>>
汽车常识全面介绍 - 悬挂系统
查看>>
电子政务方向:We7.Cloud政府云门户
查看>>
虚拟机Centos7连接Internet
查看>>
ansible 基本操作(初试)
查看>>
更改tomcat的根目录路径
查看>>
51nod 1292 字符串中的最大值V2(后缀自动机)
查看>>
加快ALTER TABLE 操作速度
查看>>