You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

232 lines
12 KiB
Markdown

2 years ago
# 10丨案例在JMeter中如何设置参数化数据
今天我们来做一个实例看下在JMeter中如何合理地设置参数化数据。
## 正式场景前的基准测试
在没有做业务混合场景之前我们需要先做Benchmark测试来确定一个登录业务能支持多少的业务量这样就可以在业务混合场景中根据场景中各业务的比例来确定登录的数据需要多少真实的数据。
我们继续用上一篇文章中用户登录的例子,单独测试登录业务,结果如下:
```
Java
summary + 125 in 00:00:04 = 31.0/s Avg: 28 Min: 0 Max: 869 Err: 0 (0.00%) Active: 1 Started: 1 Finished: 0
summary + 3404 in 00:00:30 = 113.2/s Avg: 31 Min: 0 Max: 361 Err: 0 (0.00%) Active: 6 Started: 6 Finished: 0
summary + 4444 in 00:00:30 = 148.4/s Avg: 57 Min: 0 Max: 623 Err: 10 (0.23%) Active: 11 Started: 11 Finished: 0
```
从上面的结果可以看到登录业务能达到的TPS是113左右这里我们取整为100以方便后续的计算。
## 在测试工具中配置参数
在上面的试探性测试场景中不需要观察系统的资源只需要根据TPS做相应的数据统计即可。
前面我们知道在这个示例中只做了近10万条的用户数据为了方便示例进程。
下面我们从数据库中查询可以支持登录5分钟不重复的用户数据。根据前面的公式我们需要30000条数据。
```
Java
100x5mx60s=30000条
```
接下来连接数据库取30000条数据存放到文本中如下所示
```
Java
username,password
test00001,test00001
test00002,test00002
test00003,test00003
test00004,test00004
test00005,test00005
test00006,test00006
test00007,test00007
...................
test30000,test30000
```
## 参数化配置在JMeter中的使用说明
我们将这些用户配置到测试工具的参数当中这里以JMeter的CSV Data Set Config功能为例。配置如下
![](https://static001.geekbang.org/resource/image/d4/a6/d40134621469079dd7b9de6e19165ca6.png)
在JMeter的参数化配置中有几个技术点在这里说明一下。
“Allow quoted data?”这里有两个选择分别是False和True。它的含义为是否允许带引号的数据比如说在参数化文件中有这样的数据。
```
Java
"username","password"
"test00001","test00001"
"test00002","test00002"
...................
"test30000","test30000"
```
如果有引号这个选择必须是True。如果设置为False那么我们在脚本中会看到如下的数据
```
Java
username=%22test00001%22password=%22test00001%22
```
由于设置为FalseJMeter将")转换为了%22的URL编码很显然这个数据是错的。如果选择为True则显示如下
```
Java
username=test00001password=test00001
```
这里就显示对了。
除此之外,还有如下几个功能点需要说明:
* Recycle on EOF? 这里有三个选择False、True和Edit。前两个选择非常容易理解。False是指在没有参数的时候不循环使用True是指在没有参数的时候循环使用。Edit是指在没有参数的时候会根据定义的内容来调用函数或变量。
* Stop thread on EOF?这里有三个选择False、True和Edit。含义和上面一致。
* Sharing mode : 这里有四个选择All threads、Current thread group、Current thread、Edit。
Sharing mode的前三个选择是比较容易理解的参数是在所有线程中生效在当前线程组生效还是在当前线程中生效。但这里的Edit和前两个参数中的Edit相比有不同的含义。这里选择了Edit之后会出现一个输入框就是说这里并不是给引用函数和参数使用的而是要自己明确如何执行Sharing mode。那如何来使用呢
举例来说假设我们有Thread Group 1-5 五个线程组但是参数化文件只想在Thread Group 1、3、5中使用不想在线程组2、4中使用那么很显然前面的几个选项都达不到目的这时我们就可以选择Edit选项在这里输入`SharedWithThreadGroup1and3and5`。而在其他的线程组中配置其他参数化文件。
也就是说同样的一个变量名在线程组1/3/5中取了一组数据在线程组2/4中取了另一组数据。
以上三个参数的选项可以随意组合。于是就会得到如下表。
**需要注意的是EOF是文件结束符的意思。在下面的解释中为了更符合性能测试中的术语特意解释为参数不足时。**
以上三个功能点根据参数设计得不同,会产生不同的组合,我们依次查看一下。
![](https://static001.geekbang.org/resource/image/3c/b1/3ce3106d6b079e715cce6fcbf8de72b1.png)
这个组合显然是矛盾的,没有参数时不让循环,还不让停止线程,这不是耍流氓吗?真实的结果是什么呢?当我们执行时就会发现,参数变成了这样:
```
username=%3CEOF%3E&password=%3CEOF%3E
```
服务端果然返回了:`{"flag":false,"errMsg":"账号不存在"}`。
![](https://static001.geekbang.org/resource/image/2e/6b/2ea13362d80579cbcc8f941e53ed696b.png)
这个组合中第二个选项显然是没意义的,既然参数允许重复使用了,又怎么会发生参数不足停止线程的情况呢?
![](https://static001.geekbang.org/resource/image/e5/65/e5643ea7e5dd17bb31a715e2aa5bdf65.png)
这个组合因为第一个选项为“Edit”所以变得不确定了如果在Edit的函数或变量返回为True则和第2种组合一样如果返回为False则和第1种组合一样。
![](https://static001.geekbang.org/resource/image/d4/99/d48f109b59767bd99679f4f442494899.png)
这是一个完全合情合理的组合!
![](https://static001.geekbang.org/resource/image/4b/05/4be6a8f830155a6fdfdee518c9220305.png)
同第二个组合一样,第二个选项显然没有意义。
![](https://static001.geekbang.org/resource/image/91/0c/9107142e3f0acd73eb2f350dc698a70c.png)
这个组合同样因为第一个选项为Edit所以变得不确定了如果在Edit的函数或变量返回为True则和第3种组合一样如果返回为False则和第4种组合一样。
![](https://static001.geekbang.org/resource/image/00/e0/008693e916bf65b9cdc6586afd5fcde0.png)
这个组合因为是否停止线程的不确定性会出现两种可能有可能是第1种组合也有可能是第4种组合。
![](https://static001.geekbang.org/resource/image/ed/57/ed552dbc050ec154ac7f4f1b4148e657.png)
这个组合中是否停止线程的Edit配置没有意义因为可循环使用参数所以不会发生参数不足导致线程停止的情况。
![](https://static001.geekbang.org/resource/image/37/f0/37735347120d86198df67dffbaa31ff0.png)
这是一个古怪的组合具有相当的不确定性有可能变成第1、2、4、5种组合。
下面我们再来看下其他衍生的设置组合。
![](https://static001.geekbang.org/resource/image/91/0b/9151da6424095444d56141c5e8c11b0b.jpg)
## 真实场景下的JMeter参数配置和执行结果
根据以上的描述我们先用10个用户来测试下将Stop `thread on EOF?`改为True将`Recycle on EOF?`改为False其他不变。同时将线程组中配置为1个线程循环11次。这样设置的目的是为了看在数据不足时是否可以根据规则停掉线程组。如下所示
![](https://static001.geekbang.org/resource/image/d4/a6/d40134621469079dd7b9de6e19165ca6.png)
线程组配置如下:
![](https://static001.geekbang.org/resource/image/d2/3f/d26aeda4baea18631966b15dd5084a3f.png)
执行之后,我们会在日志中看到如下信息:
```
Java
2019-09-05 22:56:30,171 INFO o.a.j.t.JMeterThread: Stop Thread seen for thread Thread Group 1 1-1, reason: org.apache.jorphan.util.JMeterStopThreadException: End of file:/Users/Zee/Downloads/user10.csv detected for CSV DataSet:CSV Data Set Config configured with stopThread:true, recycle:false
```
可以看到在参数用完又不可循环使用参数的情况下JMeter主动停止了线程。
我们延续使用上文中场景二的条件,即希望场景中每个线程的每次迭代都用不同的数据。
为了能很快地整理出实际的结果我们只使用10条数据来模拟条件设置如下
```
线程组2
线程每线程组6
参数化数据10条
```
执行完场景后,会在日志中看到如下信息:
```
Java
2019-09-07 23:24:25,585 INFO o.a.j.t.JMeterThread: Stop Thread seen for thread Thread Group 1 1-1, reason: org.apache.jorphan.util.JMeterStopThreadException: End of file:/Users/Zee/Downloads/user10.csv detected for CSV DataSet:CSV Data Set Config configured with stopThread:true, recycle:false
2019-09-07 23:24:25,452 INFO o.a.j.t.JMeterThread: Stop Thread seen for thread Thread Group 1 1-2, reason: org.apache.jorphan.util.JMeterStopThreadException: End of file:/Users/Zee/Downloads/user10.csv detected for CSV DataSet:CSV Data Set Config configured with stopThread:true, recycle:false
2019-09-07 23:24:23,406 INFO o.a.j.t.JMeterThread: Stop Thread seen for thread Thread Group 2 2-1, reason: org.apache.jorphan.util.JMeterStopThreadException: End of file:/Users/Zee/Downloads/user10.csv detected for CSV DataSet:CSV Data Set Config configured with stopThread:true, recycle:false
2019-09-07 23:24:25,517 INFO o.a.j.t.JMeterThread: Stop Thread seen for thread Thread Group 2 2-2, reason: org.apache.jorphan.util.JMeterStopThreadException: End of file:/Users/Zee/Downloads/user10.csv detected for CSV DataSet:CSV Data Set Config configured with stopThread:true, recycle:false
```
可见所有的线程都按我们的配置停止了线程,同时各线程取得参数如下表所示:
![](https://static001.geekbang.org/resource/image/6c/ab/6cd12cb6c6b0f41dec961379380beaab.png)
每次执行场景会有不同不同点是线程组1有可能执行6次而线程组2只执行4次或者线程组1中的线程2执行次数比线程1执行次数多。但总体执行次数会是10次。
如果数据可以被线程平均分配,则每个线程的迭代次数会相同。如果数据不能被线程平均分配,则每个线程的迭代次数不会相同,但相差不会大。
## 参数化配置在LoadRunner中的使用说明
在LoadRunner中参数配置页面如下
![](https://static001.geekbang.org/resource/image/88/a7/88659e50f65ba8cb8005aa2e82c742a7.png)
它的取值组合如下所示:
![](https://static001.geekbang.org/resource/image/c7/a3/c705bcf1b185248e05a0457ac0a010a3.png)
以上的组合中组合7对应着上文中JMeter真实场景中每次迭代取不同数据的组合即JMeter中的参数组合4。
## 总结
通过今天的内容,我们对性能测试中的参数化做了一次解析,在执行性能测试时,我们需要根据实际的业务场景选择不同的数据量和参数设置组合。
不同的压力工具在参数化的实现逻辑上也会不同,但是参数化必须依赖业务逻辑,而不是工具中能做到什么功能。所以在参数化之前,我们必须分析真实业务逻辑中如何使用数据,再在工具中选择相对应的组合参数的方式去实现。
这里我总结一下性能工作中参数化的逻辑,希望对你有所启发。
1. 分析业务场景;
2. 罗列出需要参数化的数据及相对应的关系;
3. 将参数化数据从数据库中取出或设计对应的生成规则;
4. 合理地将参数化数据保存在不同的文件中;
5. 在压力工具中设置相应的参数组合关系,以便实现模拟真实场景。
通过以上步骤,我们就可以合理的参数化数据,模拟出真实场景。
## 思考题
你可以思考一下下面几个问题:
1. 为什么参数化数据要符合生产环境的数据分布?
2. 为什么参数化数据要关注组合逻辑关系,而不是随意设置组合?
欢迎你在评论区写下你的思考,也欢迎把这篇文章分享给你的朋友或者同事,一起交流一下。