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}

相关推荐
sszmvb123417 小时前
测试开发 | 电商业务性能测试: Jmeter 参数化功能实现注册登录的数据驱动
jmeter·面试·职场和发展
小码哥说测试19 小时前
接口测试用例设计的关键步骤与技巧解析!
自动化测试·测试工具·jmeter·职场和发展·测试用例·接口测试·postman
小钱c71 天前
Mac下安装Apache JMeter并启动
jmeter·macos·apache
古人诚不我欺1 天前
jmeter常用配置元件介绍总结之函数助手
jmeter
川石课堂软件测试1 天前
性能测试|docker容器下搭建JMeter+Grafana+Influxdb监控可视化平台
运维·javascript·深度学习·jmeter·docker·容器·grafana
古人诚不我欺1 天前
jmeter常用配置元件介绍总结之取样器
jmeter
十叶知秋1 天前
【jmeter】jmeter的线程组功能的详细介绍
数据库·jmeter·性能测试
我非夏日1 天前
JMeter基础篇
jmeter
茶馆大橘1 天前
微服务系列五:避免雪崩问题的限流、隔离、熔断措施
java·jmeter·spring cloud·微服务·云原生·架构·sentinel
土小帽软件测试2 天前
jmeter基础01-2_环境准备-Mac系统安装jdk
java·测试工具·jmeter·macos·软件测试学习