Jmeter之Beanshell详解

一、 Beanshell概念

Beanshell:

  • BeanShell是一种完全符合Java语法规范的脚本语言,并且又拥有自己的一些语法和方法;
  • BeanShell是一种松散类型的脚本语言(这点和JS类似);
  • BeanShell是用Java写成的,一个小型的、免费的、可以下载的、嵌入式的Java源代码解释器,具有对象脚本语言特性,非常精简的解释器jar文件大小为175k。
  • BeanShell执行标准Java语句和表达式,另外包括一些脚本命令和语法。

官网:http://www.BeanShell.org/

二、Jmeter常用Bean Shell

  • 定时器:  BeanShell Timer

  • 前置处理器:BeanShell PreProcessor

  • 采样器:  BeanShell Sampler

  • 后置处理器:BeanShell PostProcessor

  • 断言:   BeanShell断言

  • 监听器:  BeanShell Listener

三、Jmeter常用内置变量

1.内置变量

BeanShell脚本中不用定义,可以直接使用的变量,常用的内置变量和方法如下:

log:写日志到控制台和jmeter.log,如log.info("XXX");
vars:操作变量

vars.get("skuId"); 从jmeter中获取${skuId}变量的值

vars.put("name","test"); 将"test"保存到${name}变量中
prev:获取前面sampler返回的信息

getResponseDataAsString(); 获取响应信息

getResponseCode(); 获取响应

更多内置变量参考:https://jmeter.apache.org

举例:

2.添加 BeanShell取样器

复制代码
log.info(".....<<<<<");
log.error("xxxx");

String myip=vars.get("ip");//获取参数值(在用户定义的变量中设置了"ip"值)
log.info(myip);//日志打印

vars.put("ip","192.168.0.0");//赋值
String myip=vars.get("ip");
log.info(myip);//日志打印

3.后置处理器的应用

复制代码
log.info("前一个返回结果为:"+prev.getResponseDataAsString());
log.info("前一个请求的状态码为:"+prev.getResponseCode())

四、BeanShell的用法实例

1.BeanShell面板上写脚本

需求:

1、调用接口获取sku信息

2、判断库存,如果库存大于500,调用buy接口购买10个商品,否则购买5个商品

复制代码
// 获取接口返回的库存值
String myStock = vars.get("p_stock");//转换为整数
int iStock = Integer.parselnt(myStock);//判断库存
if (iStock>500){
// 重新保存参数
vars.put("buyNum","10");
}else{
vars.put("buyNum","5");
}

先是get接口json提取器提取库存字段然后用到后面脚本里去做判断

2.引用外部 java源文件

引用外部源码文件然后实现md5加密完成签名接口

源码文件:

链接: https://pan.baidu.com/s/1JQlgeHGVHl8NOdSiTwNMkg?pwd=uqrs 提取码: uqrs

在beanShel中通过source("代码路径")方法引入java,然后调用方法和java一样,new一个class,再调用里面的方法

前置处理器

复制代码
//引入源代码
source("/Users/mac/Documents/study23/jmeter/md5/Md5Util.java");
//生成随机手机号
String phone = "135${__Random(10000000,99999999,myPhone)}"; 
String code = "testmay";
//生成时间戳
String time = "${__time(,myTime)}";
// 调用外部函数进行加密
String md5 = Md5Util.getMd5Hex(phone+code+time);
// 将数据另存为新的变量
vars.put("phone",phone); 
vars.put("md5",md5);

{phone} {code} ${md5}可被调用

3.调用jar包

使用beahshell调用自己写的工具类,工具类实现了密码的加、解密功能

在eclipse写好代码,然后把该类打成jar包(在类上点击右键->Export->jar file)

法一:

1、将jar包放到jmeter目录/Users/mac/Documents/apache-jmeter-5.6.2/lib/ext下

3、打开jmeter,添加一个http sampler(调用登录接口),在sampler下添加一个BeanShell PreProcessor(如果jmeter已经打开了,步骤2中jar包要生效,必须才重启jmeter

4、在beanshell PreProcessor中导入我们的jar包,调用里面的加、解密码方法,把结果保存在jmeter变量中

法二:

调用jar包

1、测试计划,Add directory or jar to classpath

2、import 所需要的类名



查看jar包内容:

进入jar路径,输入以下指令:

复制代码
jar tf testfan-md5.jar

然后调用方式用import

复制代码
import com.lee.util.Md5Util
// 生成随机手机号
String phone = "135${_Random(10000000,99999999,myPhone)}"; 
String code = "testmay";
// 生成时间戳
String time = "${__time(,myTime)}";
//调用外部函数进行加密
String md5 = Md5Util.getMd5Hex(phone+code+time);
//将数据另存为新的变量 
vars.put("phone",phone); 
vars.put("md5",md5);

4.提取 json值

需求:提取sample返回json数据中所有name字段值,返回的json格式如下:

{"body":{"apps":[{"name":"111"},{"name":"222"}]}}

jmeter中添加后置处理器BeanShell PostProcessor

说明:脚本中的导入的json包需要自己去网络下载后放到\lib\ext

链接: https://pan.baidu.com/s/1-knIb9_NulF81mIkortvoQ?pwd=h4p4 提取码: h4p4

复制代码
import org.json.*;

String response_data = prev.getResponseDataAsString();
JSONObject data_obj = new JSONObject(response_data);
String apps_str = data_obj.get("body").get("apps").toString();
JSONArray apps_array = new JSONArray(apps_str);
String[] result = new String[apps_array.length()];
for(int i=0;i<apps_array.length();i++){
    JSONObject app_obj = new JSONObject(apps_array.get(i).toString());
    String name = app_obj.get("name").toString();
    result[i] = name;
}
vars.put("result", Arrays.toString(result));

5.BeanShell断言

5.1 数据断言

内置变量

Failure:是否失败, boolean类型

FailureMessage:失败日志,在断言失败时显示

复制代码
int iStock = Integer.parselnt(vars.get("p_stock")); 
if (iStock >1500){ 
  Failure =true;
  FailureMessage ="库存数量超过了1500";
  // ResponseData是服务器返回的byte[]类型的数据
  // 如果想打印,必须转换为String类型的,用new String(ResponseData) 
  log.info(new String(ResponseData));
  //打印当前请求的url,SamplerData是String类型的数据 
  log.info(SamplerData);
}

5.2 状态断言

复制代码
//状态码断言
log.info("状态码:" + ResponseCode);
if(ResponseCode.equals("200")){ 
	Failure=false;
}
else{
	Failure=true;
	FailureMessage="响应状态码非200";  //指定失败原因
}

5.3 响应体断言

复制代码
//获取响应数据
Stringresponse= prev.getResponseDataAsString();
log.info("响应体:" + response);
//响应数据包含if(response.contains("登录成功")){
	Failure=false;
}
else{
	Failure=true;
	FailureMessage="响应数据不包含登录成功";
}

5.4 json值断言

复制代码
//JSON响应断言
import org.json.*;   //导入org.json包
Stringresponse= prev.getResponseDataAsString();  
//获取响应数据
JSONObjectresponseJson=newJSONObject(response);  
//转为JSON对象
Stringmessage= responseJson.getString("message"); 
log.info("响应message字段:" + message);
 
if(message.equals("成功")){
	Failure=false;
}
else{
	Failure=true;
	FailureMessage="响应message字段非成功";
}

6.BeanShell写数据到文件

这如果保存后就可以用到其他线程里当参数了,很实用

以后只需要更改下面的部分,这个文件就可以用了

复制代码
"csrf_token" "/Users/mac/Documents/study23/jmeter/output.txt"

需求

1、调用登录接口,获取token值

2、将token值保存到一个文件里

复制代码
String line = vars.get("csrf_token"); 
try {
    BufferedWriter writer = new BufferedWriter(new FileWriter("/Users/mac/Documents/study23/jmeter/output.txt",true)); 
    writer.write(line); 
    writer.newLine(); 
    writer.close();
} catch (IOException e) {
    e.printStackTrace();
}

运行会自动生成一个 txt文件

7.引用外部class文件

1、直接把上例中的java文件编译成class文件

进入java路径下,使用javac Md5Util.java就会编译成.class文件,同目录自动生成.class文件

【注意】macos上此处有大坑

这里有一个大坑,调用class文件时:addClassPath()方法中的路径,在macos上不能写成文件的绝对路径。这个路径只能写到包名的前一级,不能包含包名。

如果拿到的是一个单独的class文件,一定要反编译,检查包名,将这个class文件的包名层级新建出来,再将class文件放进去

2、BeanShell使用代码如下:

用addClassPath(path)方法引入 class文件,在用import导入包及类,然后就可以像java一样调用了

复制代码
//引入class文件
addClassPath("/Users/mac/Documents/study23/jmeter/md5");
//导入类名
import Md5Util
// import 包名.类名
//生成随机手机号
String phone = "135${__Random(10000000,99999999,myPhone)}"; 
String code = "testmay";
//生成时间戳
String time = "${__time(,myTime)}";
// 调用外部函数进行加密
String md5 = Md5Util.getMd5Hex(phone+code+time);
// 将数据另存为新的变量
vars.put("phone",phone); 
vars.put("md5",md5);
log.info(md5)

8.获取数据库数据并入参

jmeter数据库连接

8.1 添加配置元件 JDBC Connection Configuration

8.2 添加取样器JDBC Request

8.3 添加后置处理器 JDBC Postprocessor

复制代码
import java.util.Random;

Random random=new Random();
Object object=vars.getObject("object");  //获取sql查询结果

int size=object.size();   //获取查询结果数量
line_0 = object.get(0);   //获取object的第一个元素
filed_name = line_0.get("title");  //获取title字段的值

vars.put("size",size.toString());   //将size转换成字符串,存到变量size中
vars.put("line_0",line_0.toString());
vars.put("filed_name",filed_name.toString());

String[] fields={"title","price"};   //创建一个字符数组,里面为需要作为入参的字段

log.info(" ");
log.info("========================开始打印日志====================");
log.info("size:" + vars.get("size"));
log.info("line_0:" + vars.get("line_0"));
log.info("filed_name:" + vars.get("filed_name"));

运行

8.4 参数调用

${file_name}

${size}

相关推荐
文人sec15 小时前
性能测试-jmeter9-逻辑控制器、定时器压力并发
测试工具·jmeter·性能优化·模块测试
CesareCheung2 天前
JMeter分布式压力测试
分布式·jmeter·压力测试
测试界清流2 天前
jmeter使用技巧
jmeter
春时似衿里2 天前
jmeter配置数据库连接步骤
数据库·jmeter
新知图书2 天前
JMeter的安装部署
jmeter
程序员杰哥2 天前
什么是Jmeter? Jmeter工作原理是什么?
自动化测试·软件测试·python·测试工具·jmeter·职场和发展·测试用例
乐神嘎嘎嘎2 天前
Jmeter测试
jmeter
卓码软件测评2 天前
第三方软件测试机构【性能测试工具用LoadRunner还是JMeter?】
java·功能测试·测试工具·jmeter·性能优化
BatyTao3 天前
Jmeter执行数据库操作
数据库·jmeter
二宝哥3 天前
性能测试工具Jmeter之java.net.BindException: Address already in use
jmeter