jmeter学习记录--05--Beanshell2
学习beanshell时有不少的例子、遇到不少问题。在此记录下。
测试实例列表
A1:使用Beanshell请求作为测试请求
一个打包的Jar包,直接对其内的方法进行测试。
第一步:将接口jar包要放到%JMETER_HOME%\lib目录
第二步:编写请求。
A2:共享线程变量
JMeter中线程间共享变量可以通过定义属性值来完成,JMeter启动时会读取一些属性文件,比如jmeter.properties、user.properties,这些属性值是可以增加的,也可以修改的,通过BeanShell可以对其进行更改。
定义变量:以BeanShell Sampler为例,在其中通过props.put()来增加属性,props.get()来获取属性。如:props.put("param_a","root");
使用变量:在后续请求如Java请求中通过 ${__property(【属性名称】,,)}来获取。如:${__property(param_a,,)} 来使用。
示例:
在线程组1中使用__setProperty定义属性值(全局变量),将所需变量值${param_A}设置为属性值。
String token = bsh.args[];//输入值为assfd
//第一种写法,测试时token与${token}两种写法都无法调用变量
${__setProperty(param_A,${token},)};
//第二种写法
props.put("param_B",token);
debug请求中,打开属性与变量。查看结果树中,属性值结果为:
param_A=${token}
param_B=assfd
在线程组2,beanshell中输入
${__property(param_A,GROUP2_A,)};
${__property(param_B,GROUP2_B,)};
debug请求中,打开属性与变量。查看结果树中,变量值结果为:
GROUP2_A=${token}
GROUP2_B=assfd
疑问:1)${__setProperty(param_A,token,)}; 这里token是变量,但是实际是将这个字符串拷贝存到param_A属性里的了。换成${token}也是如此。
2)props.put添加的属性值,不会写到jmeter.properties文件里,只在当前测试计划中生效。比如当前加了一个属性A,执行过后在当前测试计划中会一直存在,即使beanshell文件中已无这个属性值的定义也是如此。重新打开一个测试计划,属性A就不存在了。
A3:使用 BeanShell 断言判断用例成功与否
用beanshell做断言结果判断
log.info(SampleResult.getResponseDataAsString());
boolean result = false;
String uid = vars.get("uid");
String token = vars.get("token");
if (SampleResult.getResponseCode().equals("200") && token.contains(uid)) {
result = true;
} else {
result = false;
}
SampleResult.setSuccessful(result);
http://www.cnblogs.com/xxyBlogs/p/5966194.html
A4:处理请求或结果,如对base64加解密,URL转码等(jmeter 3.2)
base64加解密,使用函数助手中的两个函数。
在beanshell中添加
//加密,encode为加密后字符串变量
${__base64Encode(“haishizi”,encode)};
//解密,对加密胡变量做解密处理
${__base64Decode(${encode},decode)};
debug sampler对应测试结果为:
decode="haishizi"
encode=ImhhaXNoaXppIg==
如果加密函数中${__base64Encode(“haishizi”,encode)};修改为${__base64Encode(haishizi,encode)};
对应测试结果为:
decode=haishizi
encode=aGFpc2hpemk=
A5:自定义时间格式 Jmeter-BeanShell的使用介绍
普通方式获取时间戳
在Jmeter中,可以利用${__time(,)}时间戳函数来获取十位的时间戳:
{
"time":"${__time(yyyy-MM-dd-HH-mm-ss,)}"
}

直接执行后,在“查看结果树”中显示的是:

BeanShell自定义
但是在具体使用中,有时候需要传送较为复杂的时间格式,如“2016-07-31T21:24:07.581Z”此时不能直接调用time函数,因此可以利用BeanShell获取当前时间。
import java.util.*;
import java.text.SimpleDateFormat; String str1 = (new SimpleDateFormat("yyyy-MM-dd")).format(new Date());
String str2 = (new SimpleDateFormat("HH:mm:ss.SSS")).format(new Date());
String str3 = (new SimpleDateFormat("HH:mm:ss")).format(new Date()); vars.put("str_A",str1+"T"+str2+"");
vars.put("str_B",str1+"T"+str3+".000"+""); log.info("\n"+str_A);//直接获取不生效,日志打印为void
log.info("\n"+${str_A});//直接获取不生效,日志打印为空
若使用该程序段,在Jmeter中调用变量${str_A}可以获得yyyy-MM-ddTHH:mm:ss.SSSZ格式的时间,调用变量${str_B}可以获得yyyy-MM-ddTHH:mm:ssZ格式的时间。
http请求中输入,选择post方法。
{
"time":"${__time(yyyy-MM-dd-HH-mm-ss,)}"
"time2":${str_A}
"time3":${str_B}
"time4":${str1}
}
查看结果树中结果为:
POST data:
{
"time":"2017-07-26-18-23-56"
"time2":2017-07-26T18:23:56.8692
"time3":2017-07-26T18:23:56.0002
"time4":${str1}
}
疑问:为何str1字符串直接日志输出或使用变量不可以,将其转化为内置变量就可正常使用?
A6:把测试结果放到一个excel文件中

FileWriter fstream = new FileWriter("d:\\a.csv",true);
BufferedWriter out = new BufferedWriter(fstream);
out.write("uid"+","+vars.get("uid")+",");
out.write("mobileNumber"+","+vars.get("mobileNumber")+",");
out.write("inviteCode"+","+vars.get("inviteCode")+",");
out.write("\n");
//out.write(vars.get("inviteCode"));
//out.write(System.getProperty("line.separator"));
out.close();
fstream.close();
A7: 对redis的操作
import redis.clients.jedis.Jedis;
try{
Jedis jedis = new Jedis("${host}",${port},10000);
if (jedis != null) {
System.out.println("connect to redis server sucessfully");
}
String hostCount = jedis.get("api_index_ACount");
String videoCount = jedis.get("api_index_BCount");
System.out.println(ACount);
System.out.println(BCount);
vars.put("hostCount",ACount);
vars.put("videoCount",BCount);
jedis.close();
}
catch(Exception e){
System.out.println(e);
}
sampler 请求
import redis.clients.jedis.Jedis;
Jedis jedis = new Jedis("${host}",${port},10000);
String ACount = jedis.get("api_index_ACount");
String BCount = jedis.get("api_index_BCount");
jedis.close();
String response_data = prev.getResponseDataAsString();
if(response_data.contains(ACount)&&response_data.contains(BCount)){
FailureMessage="success";
}
else{
Failure=true;
FailureMessage="fail:"+ACount+""+BCount;
}
断言
问题列表
Q1:自定义函数的变量赋值时报错(jmeter 3.2)---未解决
变量定义有3种情况:
第一种:param_A,直接通过vars.put来定义参数值;
第二种:param_B,通过bsh.args[0]获取到传入的参数值123,将其赋值。
第三种:通过函数变量来实现,将字符串abc变量的值赋值给unique_id。

import java.util.UUID;
vars.put("param_A","123456");
vars.put("param_B",bsh.args[0]);
UUID uuid1 = UUID.randomUUID();
String abc = uuid1.toString();
vars.put("unique_id",abc);
log.info("\n_abc_value:"+abc);
log.info("\n unique_id value:"+${unique_id});
beanshell变量定义

result = "\n"+"param_A:"+bsh.args[0]+"\n"; //1:从参数框中获取参数变量,不适用
result += "param_B:"+bsh.args[1]+"\n";
result += "unique_id:"+bsh.args[2]+"\n"; ResponseCode = 500;//回写请求响应代码,借鉴的http响应码
ResponseMessage = "test ok!!";//回写响应信息
IsSuccess = true;//回写响应结果 log.info("\nlog test:"+result);//调用jmeter内置log对象,日志会打印到jmeter.log中
SampleResult.setResponseData("SampleResult:"+result+"\n");//回写响应数据,SampleResult为jmeter内置回写结果对象,有很多可用方法,可以参看源码
return "return:"+result;
beanshell使用变量

{unique_id}值并未取实际的参数值。看log日志:

2017-07-12 09:00:53,939 ERROR o.a.j.u.BeanShellInterpreter: Error invoking bsh method: eval Sourced file: inline evaluation of: ``import java.util.UUID; vars.put("param_A","123456"); vars.put("param_B",bsh.arg . . . '' : Attempt to access property on undefined variable or class name
可看到abc字符串取值还是正常的,但是将其赋值unique_id后报错。
vars:即JMeterVariables,操作jmeter变量,这个变量实际引用了JMeter线程中的局部变量容器(本质上是Map),常用方法:
a) vars.get(String key):从jmeter中获得变量值
b) vars.put(String key,String value):只能用String类型的参数,数据存到jmeter变量中,其作用可简单理解为赋值操作:key=value,更多方法可参考:org.apache.jmeter.threads.JMeterVariables
Map<String, String> paramMap = new HashMap<String, String>();
其中有这行代码时总会提示:
2019-01-30 16:04:54,702 ERROR o.a.j.u.BeanShellInterpreter: Error invoking bsh method: eval In file: inline evaluation of: ``import org.apache.commons.codec.digest.DigestUtils; //导入md5加密的包 import java. . . . '' Encountered "," at line 12, column 11.
2019-01-30 16:04:54,702 WARN o.a.j.p.j.s.BeanShellSampler: Exception executing script. org.apache.jorphan.util.JMeterException: Error invoking bsh method: eval In file: inline evaluation of: ``import org.apache.commons.codec.digest.DigestUtils; //导入md5加密的包 import java. . . . '' Encountered "," at line 12, column 11.
将这行修改为:Map paramMap = new HashMap(); 后解决
后续用put方法将信息存到paramMap中。但是如果引入的jar中有代码也是使用的map传参,而beanshell中如果调用这个方法时也会报错。
beanshell调用:
String sign = MD5Util.proSign(paramMap,secret); jar包中源码:
public String proSign(Map<String, String> params, String secret)
报错:
2019-01-30 16:38:53,914 ERROR o.a.j.u.BeanShellInterpreter: Error invoking bsh method: eval Sourced file: inline evaluation of: ``import com.chain.*; import java.util.Map; import java.util.HashMap; import java . . . '' : Typed variable declaration : Cannot reach instance method: proSign( java.util.Map, java.lang.String ) from static context: com.test.MD5Util
2019-01-30 16:38:53,914 WARN o.a.j.p.j.s.BeanShellSampler: Exception executing script. org.apache.jorphan.util.JMeterException: Error invoking bsh method: eval Sourced file: inline evaluation of: ``import com.test.*; '' : Typed variable declaration : Cannot reach instance method: proSign( java.util.Map, java.lang.String ) from static context: com.test.MD5Util
这种调用方式,没有解决。用其他方式暂时规避了。
参考:
jmeter学习记录--05--Beanshell2的更多相关文章
- jmeter学习记录--10--二次开发环境搭建
JMeter源码集成到Eclipse.JMeter二次开发(1)-eclipse环境配置及源码编译 ,根据此文章记录将jmeter源码集成到myecplise 第一步:下载jmeter源码http:/ ...
- jmeter学习记录--09--命令行运行与生成报告
一. 使用命令行方式运行Jmeter 1.1 为什么 使用GUI方式启动jmeter,运行线程较多的测试时,会造成内存和CPU的大量消耗,导致客户机卡死. 所以正确的打开方式是在GUI模式下调 ...
- jmeter学习记录--04--Beanshell
一.什么是Bean Shell BeanShell是一种完全符合Java语法规范的脚本语言,并且又拥有自己的一些语法和方法;BeanShell是一种松散类型的脚本语言(这点和JS类似); BeanSh ...
- jmeter学习记录--08--第三方测试组件
我们安装的JMeter版本,功能仍然有欠缺,插件是一种补充,官方提供了很多插件. 官网地址:http://www.jmeter-plugins.org/, 里面有很多可以安装到JMeter的插件,基本 ...
- jmeter学习记录--07--jmeter元件
通过jmeter元件可以模拟负载.参数化.设置关联.设置检查点.设置集合点.控制场景运行.监控测试结果等. 1.逻辑控制器:比如foreach控制器,查询到了订单并要对每个订单进行出库操作,以订单号作 ...
- jmeter学习记录--03--jmeter负载与监听
jmeter场景主要通过线程组设置完成,有些复杂场景需要与逻辑控制器配合. 一.测试计划设计与执行 场景设计 jmete线程组实际是一个线程池,根据用户设置进行线程池的初始优化,在运行时做各种异常的处 ...
- Jmeter学习记录
JSON正则表达式提取规则 https://www.cnblogs.com/hc1020/p/7723720.html Jmeter非GUI下执行日志 执行命令 ./jmeter -n -t $ ...
- JMeter学习记录收藏
1.如何进行一个简单的性能测试 2.JMeter各种功能名词解释,比较全 3.聚合报告分析 4.CSV文件参数化,名词解释 5.JMeter快捷键
- Python学习记录day6
title: Python学习记录day6 tags: python author: Chinge Yang date: 2016-12-03 --- Python学习记录day6 @(学习)[pyt ...
随机推荐
- java基础(十八)----- java动态代理原理源码解析
关于Java中的动态代理,我们首先需要了解的是一种常用的设计模式--代理模式,而对于代理,根据创建代理类的时间点,又可以分为静态代理和动态代理. 静态代理 1.静态代理 静态代理:由程序员创建或特定工 ...
- vue-cli项目使用mock数据的方法(借助express)
前言 现如今前后端分离开发越来越普遍,前端人员写好页面后可以自己模拟一些数据进行代码测试,这样就不必等后端接口,提高了我们开发效率.今天就来分析下前端常用的mock数据的方式是如何实现的. 主体 项目 ...
- 机器学习之支持向量机—SVM原理代码实现
支持向量机—SVM原理代码实现 本文系作者原创,转载请注明出处:https://www.cnblogs.com/further-further-further/p/9596898.html 1. 解决 ...
- Python3+Selenium2完整的自动化测试实现之旅(一):自动化测试环境搭建
1 环境搭建准备 (1) 下载Python3版本的安装包,直接官网下载即可:Python官网:https://www.python.org/ (2) 下载Python的基础工具包p ...
- JS 数组、对象的深拷贝
博客地址:https://ainyi.com/72 JavaScript 程序中,对于简单的数字.字符串可以通过 = 赋值拷贝 但是对于数组.对象.对象数组的拷贝,就有浅拷贝和深拷贝之分 浅拷贝就是当 ...
- 【Oracle学习笔记】索引
1 简介 1)索引是数据库对象之一,用于加快数据的检索,类似于书籍的索引.在数据库中索引可以减少数据库程序查询结果时需要读取的数据量,类似于在书籍中我们利用索引可以不用翻阅整本书即可找到想要的信息. ...
- 数据结构——Java实现链栈
一.分析 栈是限定仅在表的一端进行插入或删除操作的线性表,对于栈来说,操作端称为栈顶,另一端则称为栈底,栈的修改是按照后进先出的原则进行的,因此又称为后进先出的线性表. 链栈是指采用链式存储结构实现的 ...
- OO_BLOG2_多线程电梯模拟
作业2-1 单部多线程傻瓜调度(FAFS)电梯的模拟 I. 基于度量的程序结构分析 1)程序结构与基本度量统计图 2)分析 这次作业基本奠定了本人三次电梯作业的基本架构,简述如下: Elevato ...
- C# CAD批量转换为图片
最近写了个工具,将指定目录下的CAD文件批量转换为图片格式. 首先需要添加对应的引用 : 在AutoCAD2008的环境下对应AutoCAD 2008 Type Library 和 AutoCAD/O ...
- 查看apk签名 和 keystore 的信息
原文出处:https://www.jianshu.com/p/90b698002215 1.keytool -printcert -file ***(把apk文件下的META- INF文件夹解压出来, ...