测试工具
JMeter 4.0
场景描述
- 公司是某电商网站,测试Search框在用户可以输入关键字后,返回一些相应的关键字或者商品信息等。
- 用户在输入字符时,每隔0.5秒停顿一次,直到字符输入完毕;
- 用户输入的每一个字符都会触发JS请求Service进行处理,并返回结果;
- 长短皆有,有空格和特殊符号;
测试场景
由提供一些关键字作为测试数据;
Web Server & Service端
常规场景
- 输入这些关键字,查看其返回的结果;使用用户最终输入的完整关键字,获取性能指标;(其作用是模拟用户使用粘贴复制的方式,查询的用户行为)
- 精确模拟用户行为,当用户每次输入关键字的字符时,发送请求产生大压力下的情况。并且在每次输入字符后,使用高斯分布计时器,延迟0.5秒,偏差值为0.1秒的方式来模拟。脚本编写时,外层使用Transaction controller,内层使用Foreach controller的组合来模拟这一行为方式;
- 采用20,100 User不同负载下持续10 min的测试方法
异常场景
- 增压测试;(不断增加压力,直至达到性能极限)
- 定时器定点发送请求;(不断加压,使用类似于秒杀活动场景来检测服务器)
测试目的
- 通过性能测试建立性能基线;
- 监控服务器在较大压力下的工作情况,查找是否有隐藏的弱点;
- 分析测试结果数据,确保服务器集群的工作负荷预期需求;
- 为线上服务维护或异常处理时的策略提供依据;
脚本编写
为了模拟用户行为所编写的脚本较为复杂,里面涉及:
- BeanShell的脚本编写,由于还是有java基础,所以编写本身并不难;
- 计时器的应用,使用高斯计时器(正态分布计时器)
- 使用Transaction Controller和ForEach Controller进行测试;
这里面有几个小坑:
- 脚本编写思路是将keyword得到之后,将其按照长度分割成若干子串:
以keyword+key为例:
其分解成:
k
ke
key
keyw
keywo
keywor
keyword
keyword+
keyword+k
keyword+ke
keyword+key
因此需要使用foreach controller来包含HTTP Request Sampler。
注意,但外层的BeanShell脚本编写不能使用BeanShell PreProcesser来编写,因为这样它就不会使内层的HTTP Request工作,原因不明等以后调试源代码才清楚,所以请使用BeanSampler来做这个事情。
-
在代码中如果使用字符串赋值,涉及到它本身的变量使用的,需要在变量上加双引号。
如图:
如上图中的方法keyWordLength()是个声明;如果需要调用就必须在脚本的下方书写keyWordLength();否则脚本并没有运行;
- 通过阅读Beanshell Sampler中的代码可以知道,我最终需要的变量名为keyWd_{子字符串下标},这个在jmeter的控制器中,只能使用ForEach Controller。它有三个变量:
- Input variable prefix,让你使用变量名的下标;
- Start index for loop (exclusive),这个就是你下标开始的位置,但是特别坑的是,这个值是被排除在外的,所以如果是从0开始,我只能写-1;
- End index for loop (inclusive),这个就是下标的最后一位,按照脚本中设置的值,我应该写${keyWdLength}-1;
- Add "_" before number? 这个请勾选,在未知长度的循环中,这个控制器显得那么得雪中送炭;
-
因为Foreach controller中为变量自动加下划线,就省去了变量拼接的问题。实际上,没有这个控件可能要写更复杂的脚本才行。
- 由于关键字存在空格,因此我一开始将关键字中的空格统一转成了%20。但是在实际请求时:
HttpClient会报错java.net.URISyntaxException: Malformed escape pair at index XX....;
原因在于URL中的%是特殊符号,我在脚本中的写法是将关键字逐一拼接,因此会有一个关键字刚好是只带有%,这是非法的。因此,才会报错。
解决方法:在上图中的Parameters页中可以勾选“Encode?”。
- 把图中的Simple Controller改成Transaction Controller就可以了。在较复杂场景中,报告中生成TPS结果是更有价值的数据。
剩下的脚本就比较简单了,我就不再赘述。