当前位置:   article > 正文

Jmeter之Beanshell详解

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取样器

  1. log.info(".....<<<<<");
  2. log.error("xxxx");
  3. String myip=vars.get("ip");//获取参数值(在用户定义的变量中设置了“ip”值)
  4. log.info(myip);//日志打印
  5. vars.put("ip","192.168.0.0");//赋值
  6. String myip=vars.get("ip");
  7. log.info(myip);//日志打印

3.后置处理器的应用

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

四、BeanShell的用法实例

1.BeanShell面板上写脚本

需求: 

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

  1. // 获取接口返回的库存值
  2. String myStock = vars.get(“p_stock");//转换为整数
  3. int iStock = Integer.parselnt(myStock);//判断库存
  4. if (iStock>500){
  5. // 重新保存参数
  6. vars.put("buyNum","10");
  7. }else{
  8. vars.put(“buyNum","5");
  9. }

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

2.引用外部 java源文件

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

源码文件:

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

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

前置处理器

  1. //引入源代码
  2. source("/Users/mac/Documents/study23/jmeter/md5/Md5Util.java");
  3. //生成随机手机号
  4. String phone = "135${__Random(10000000,99999999,myPhone)}";
  5. String code = "testmay";
  6. //生成时间戳
  7. String time = "${__time(,myTime)}";
  8. // 调用外部函数进行加密
  9. String md5 = Md5Util.getMd5Hex(phone+code+time);
  10. // 将数据另存为新的变量
  11. vars.put("phone",phone);
  12. 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 

  1. import com.lee.util.Md5Util
  2. // 生成随机手机号
  3. String phone = "135${_Random(10000000,99999999,myPhone)}";
  4. String code = "testmay";
  5. // 生成时间戳
  6. String time = "${__time(,myTime)}";
  7. //调用外部函数进行加密
  8. String md5 = Md5Util.getMd5Hex(phone+code+time);
  9. //将数据另存为新的变量
  10. vars.put("phone",phone);
  11. 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

  1. import org.json.*;
  2. String response_data = prev.getResponseDataAsString();
  3. JSONObject data_obj = new JSONObject(response_data);
  4. String apps_str = data_obj.get("body").get("apps").toString();
  5. JSONArray apps_array = new JSONArray(apps_str);
  6. String[] result = new String[apps_array.length()];
  7. for(int i=0;i<apps_array.length();i++){
  8. JSONObject app_obj = new JSONObject(apps_array.get(i).toString());
  9. String name = app_obj.get("name").toString();
  10. result[i] = name;
  11. }
  12. vars.put("result", Arrays.toString(result));

5.BeanShell断言

5.1 数据断言 

内置变量
Failure:是否失败, boolean类型
FailureMessage:失败日志,在断言失败时显示

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

5.2 状态断言 

  1. //状态码断言
  2. log.info("状态码:" + ResponseCode);
  3. if(ResponseCode.equals("200")){
  4. Failure=false;
  5. }
  6. else{
  7. Failure=true;
  8. FailureMessage="响应状态码非200"; //指定失败原因
  9. }

5.3 响应体断言

  1. //获取响应数据
  2. Stringresponse= prev.getResponseDataAsString();
  3. log.info("响应体:" + response);
  4. //响应数据包含if(response.contains("登录成功")){
  5. Failure=false;
  6. }
  7. else{
  8. Failure=true;
  9. FailureMessage="响应数据不包含登录成功";
  10. }

5.4 json值断言 

  1. //JSON响应断言
  2. import org.json.*; //导入org.json包
  3. Stringresponse= prev.getResponseDataAsString();
  4. //获取响应数据
  5. JSONObjectresponseJson=newJSONObject(response);
  6. //转为JSON对象
  7. Stringmessage= responseJson.getString("message");
  8. log.info("响应message字段:" + message);
  9. if(message.equals("成功")){
  10. Failure=false;
  11. }
  12. else{
  13. Failure=true;
  14. FailureMessage="响应message字段非成功";
  15. }

6.BeanShell写数据到文件

这如果保存后就可以用到其他线程里当参数了,很实用
以后只需要更改下面的部分,这个文件就可以用了

"csrf_token" "/Users/mac/Documents/study23/jmeter/output.txt"

需求
1、调用登录接口,获取token值
2、将token值保存到一个文件里

  1. String line = vars.get("csrf_token");
  2. try {
  3. BufferedWriter writer = new BufferedWriter(new FileWriter("/Users/mac/Documents/study23/jmeter/output.txt",true));
  4. writer.write(line);
  5. writer.newLine();
  6. writer.close();
  7. } catch (IOException e) {
  8. e.printStackTrace();
  9. }

运行会自动生成一个 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一样调用了

  1. //引入class文件
  2. addClassPath("/Users/mac/Documents/study23/jmeter/md5");
  3. //导入类名
  4. import Md5Util
  5. // import 包名.类名
  6. //生成随机手机号
  7. String phone = "135${__Random(10000000,99999999,myPhone)}";
  8. String code = "testmay";
  9. //生成时间戳
  10. String time = "${__time(,myTime)}";
  11. // 调用外部函数进行加密
  12. String md5 = Md5Util.getMd5Hex(phone+code+time);
  13. // 将数据另存为新的变量
  14. vars.put("phone",phone);
  15. vars.put("md5",md5);
  16. log.info(md5)

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

jmeter数据库连接 

8.1 添加配置元件 JDBC Connection Configuration

8.2 添加取样器JDBC  Request

8.3 添加后置处理器 JDBC Postprocessor

  1. import java.util.Random;
  2. Random random=new Random();
  3. Object object=vars.getObject("object"); //获取sql查询结果
  4. int size=object.size(); //获取查询结果数量
  5. line_0 = object.get(0); //获取object的第一个元素
  6. filed_name = line_0.get("title"); //获取title字段的值
  7. vars.put("size",size.toString()); //size转换成字符串,存到变量size
  8. vars.put("line_0",line_0.toString());
  9. vars.put("filed_name",filed_name.toString());
  10. String[] fields={"title","price"}; //创建一个字符数组,里面为需要作为入参的字段
  11. log.info(" ");
  12. log.info("========================开始打印日志====================");
  13. log.info("size:" + vars.get("size"));
  14. log.info("line_0:" + vars.get("line_0"));
  15. log.info("filed_name:" + vars.get("filed_name"));

运行

8.4 参数调用

${file_name} 

${size}

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/菜鸟追梦旅行/article/detail/514522
推荐阅读
相关标签
  

闽ICP备14008679号