线程组

线程组作为JMeter测试计划的核心组件之一,对于模拟并发用户的行为至关重要。线程组元件是整个测试计划的入口,所有的取样器和控制器必须放置在线程组下。

可以将线程组视为一个虚拟用户池,其中每个线程可被理解为一个虚拟用户,多个虚拟用户同时执行相同的一批任务。

在这个虚拟用户池中,每个线程之间是相互隔离且互不影响的。每个线程的执行过程中,操作的变量不会对其他线程的变量值产生影响。

线程组的关键任务之一是定义并发用户的行为,包括设置线程数、循环次数、启动延迟等关键参数。通过适当配置线程组,测试人员可以模拟多用户在系统中同时执行任务的场景,从而评估系统的性能和稳定性。

通过灵活使用setup线程组、线程组、tearDown线程组、开放模型线程组,配置前置操作、主要操作、后置操作,更能真实、详细的评估系统。

线程组分为四类:

  • 线程组
  • setUp线程组
  • tearDown线程组
  • 开放模型线程组

线程组、setUp线程组、tearDown线程组控制面板中的元素基本一致:

  • 名称、注释
  • 在取样器错误后执行的动作
  • 线程数
  • Ramp-Up时间
  • Same user on each iteration
  • 延迟创建线程直到需要(只有线程组有)
  • 调度器

开放模型线程组控制面板中的元素:

  • 名称、注释
  • 在取样器错误后执行的动作
  • 调度计划
  • 随机种子

取样器错误后执行的动作

在JMeter中,取样器(Sampler)是用于模拟用户请求发送到目标服务器的组件,例如HTTP请求、FTP请求等。当取样器执行过程中出现错误时,可以通过配置相应的动作来处理这些错误。以下是一些处理取样器错误时,线程组中常见方式:

  • 停止线程

    任何一个线程(用户)在执行过程中遇到错误时,该线程被停止,不影响其他线程(用户)。

  • 启动下一进程循环

    任何一个线程(用户)在执行过程中遇到错误时,Jmeter会立即停止当前线程的本次执行,并进行当前线程(用户)的下次执行,主要应用于线程多次循环时。

  • 继续(无需演示)

    JMeter将在取样器执行错误时,忽略错误继续执行本线程的后续操作及执行其他线程。

停止线程-多线程


示例接口代码

@ThreadGroup.route('/api/ThreadGroup5/', methods=['GET', 'POST'])
def threadgroup5():
return '200'

示例Jmeter脚本

  • 测试计划下添加线程组

    取样器错误后执行的动作中勾选停止线程

    线程数3

  • 线程组下依次添加2个HTTP 请求取样器

    名称:错误请求-${yonghu}(在前)、正确请求-${yonghu}

    请求地址:HTTP://127.0.0.1:5000/api/ThreadGroup5/

    请求方式:GET

  • 线程组下添加CSV 数据文件设置(右键-添加-配置元件)

    文件名ceshi.txt的路径

     ceshi.txt文件内容:(复制后,手动删除前面的空格)

      200,用户1

      1111,用户2

      200,用户3

    文件编码UTF-8

    变量名称ceshi,yonghu

  • 错误请求取样器下添加响应断言

    值:${ceshi}

  • 测试计划中,添加查看结果树

运行结果

连续运行了3次,结果是一致的。总共有三个用户执行线程组,其中用户1用户3完全执行成功;用户2只执行了错误请求

因为设置取样器错误后执行的动作停止线程用户2执行错误请求时发生错误,Jmeter只会停止用户2的后续执行,不会影响其他线程。

多线程组也是多线程,读者在实际的脚本编写中,要注意每个线程的情况去使用停止线程

停止线程-多循环


示例接口代码

@ThreadGroup.route('/api/ThreadGroup5/', methods=['GET', 'POST'])
def threadgroup5():
return '200'

示例Jmeter脚本

  • 测试计划下添加线程组

    取样器错误后执行的动作中勾选停止线程

    循环次数3

  • 线程组下依次添加2个HTTP 请求取样器

    名称:错误请求-${xunhuan}(在前)、正确请求-${xunhuan}

    请求地址:HTTP://127.0.0.1:5000/api/ThreadGroup5/

    请求方式:GET

  • 线程组下添加CSV 数据文件设置(右键-添加-配置元件)

    文件名ceshi.txt的路径

     ceshi.txt文件内容:(复制后,手动删除前面的空格)

      200,第1次循环

      1111,第2次循环

      200,第3次循环

    文件编码UTF-8

    变量名称ceshi,xunhuan

  • 错误请求取样器下添加响应断言

    值:${ceshi}

  • 测试计划中,添加查看结果树

运行结果

连续运行了3次,结果是一致的。用户在第2次循环执行到错误请求时,Jmeter停止测试。

因为设置取样器错误后执行的动作停止线程,用户的第2次循环,执行错误请求时发生错误,Jmeter停止用户的后续执行(就它一个线程)。

启动下一进程循环


示例接口代码

@ThreadGroup.route('/api/ThreadGroup5/', methods=['GET', 'POST'])
def threadgroup5():
return '200'

示例Jmeter脚本

  • 测试计划下添加线程组

    取样器错误后执行的动作中勾选启动下一线程循环

    循环次数3

  • 线程组下依次添加2个HTTP 请求取样器

    名称:错误请求-${xunhuan}(在前)、正确请求-${xunhuan}

    请求地址:HTTP://127.0.0.1:5000/api/ThreadGroup5/

    请求方式:GET

  • 线程组下添加CSV 数据文件设置(右键-添加-配置元件)

    文件名ceshi.txt的路径

     ceshi.txt文件内容:(复制后,手动删除前面的空格)

      200,第1次循环

      1111,第2次循环

      200,第3次循环

    文件编码UTF-8

    变量名称ceshi,xunhuan

  • 错误请求取样器下添加响应断言

    值:${ceshi}

  • 测试计划中,添加查看结果树

运行结果

连续运行了3次,结果是一致的。用户执行了3次循环,其中第2次循环中,错误请求出现错误,跳过正确请求

因为设置取样器错误后执行的动作启动下一线程循环,用户的第2次循环,执行错误请求时发生错误,Jmeter会跳过用户的本次执行,进行用户的后续执行。

ramp-up时间


ramp-up时间用于设置启动所有线程所需要的时间。例如:线程数设置为10,ramp-up时间设置为100秒,那么JMeter将使用100秒使10个用户启动并运行,即每个用户将在前一个用户启动后的10秒启动。

如果ramp-up值设置得很小、线程数又设置得很大,刚开始执行测试时会对服务器产生很大的压力。

示例接口代码

@ThreadGroup.route('/api/ThreadGroup5/', methods=['GET', 'POST'])
def threadgroup5():
return '200'

示例Jmeter脚本

  • 测试计划下添加线程组

    Ramp-Up时间:9

    线程数3

  • 线程组下添加1个HTTP 请求取样器

    请求地址:HTTP://127.0.0.1:5000/api/ThreadGroup5/

    请求方式:GET

  • 测试计划中,添加查看结果树

运行结果

连续运行了3次,结果是一致的。3个用户执行线程组,各个用户的请求时间分别为2024-04-15 16:12:37 CST2024-04-15 16:12:40 CST2024-04-15 16:12:43 CST

3个用户执行请求的间隔时间正好是3秒,即ramp-up时间/线程数

same user on each iteration(在每次迭代中使用相同的用户)


没有研究出来它有什么用。经过我的测试,same user on each iteration(在每次迭代中使用相同的用户)启用与否,作用是一样的。

如读者对此有不同见解,欢迎与我联系,共同探讨。目前,我十分费解。

延迟创建线程直到需要


当在JMeter中启用延迟创建线程直到需要时,JMeter会根据预设的Ramp-up时间动态地分配线程。假设Ramp-up时间设置为20秒,线程数为10,那么JMeter会在测试启动后立即创建第一个线程并开始请求处理。随后,每隔2秒,JMeter将创建下一个线程,直到所有线程都被启动。

如果关闭“延迟创建线程直到需要”选项,JMeter会在测试开始时一次性创建所有线程。使用同样的参数,即在测试一开始,JMeter会立即创建全部的10个线程。这些线程会按照设定的“Ramp-up时间”进行执行,每个线程将间隔2秒启动。

延迟创建线程直到需要这一配置的目的是为了应对测试机性能有限的情况。通过这种方式,可以避免在测试初期就创建所有线程,导致资源过度占用和可能的性能问题。这种方法有助于平滑地增加系统负载,同时防止资源瞬间紧张导致测试无法正常进行。

调度器-启动延迟


调度器主要控制线程操作时间。启用调度器后,可以输入持续时间(值不能为空),启动延迟来控制线程组的操作时间及线程组操作前的延迟时间。

同时输入持续时间启动延迟时,先计算启动延迟,再计算持续时间

示例接口代码

@ThreadGroup.route('/api/ThreadGroup5/', methods=['GET', 'POST'])
def threadgroup5():
return '200'

示例Jmeter脚本

  • 测试计划下添加线程组

    启用调度器

    持续时间:10

    启动延迟3

  • 线程组下添加1个HTTP 请求取样器

    请求地址:HTTP://127.0.0.1:5000/api/ThreadGroup5/

    请求方式:GET

  • 测试计划中,添加查看结果树

运行结果

注意看图中右上角-黄色三角形左边的计时器,值固定在2秒。这个计时器计算整个测试计划的持续时间。由于示例请求的接口响应较快,可以理解为计时器的时间就是HTTP请求时的时间。

因为启动延迟设置为3秒,所以HTTP请求会在延迟3秒执行。不过计时器的时间是2秒,误差1秒。我多次试过把持续时间启动延迟的时间拉长,误差还是1秒。

调度器-持续时间


调度器主要控制线程操作时间。启用调度器后,可以输入持续时间(值不能为空),启动延迟来控制线程组的操作时间及线程组操作前的延迟时间。

同时输入持续时间启动延迟时,先计算启动延迟,再计算持续时间

示例接口代码

@ThreadGroup.route('/api/ThreadGroup6/', methods=['GET', 'POST'])
def threadgroup6():
sleep(3)
return '200'

示例Jmeter脚本

  • 测试计划下添加线程组

    启用调度器

    持续时间:2

  • 线程组下添加1个HTTP 请求取样器

    请求地址:HTTP://127.0.0.1:5000/api/ThreadGroup6/

    请求方式:GET

  • HTTP 请求下添加1个固定定时器

    值:3000

  • 测试计划中,添加查看结果树

运行结果

图中结果树中什么都没有,这是因为线程组的持续时间只有2秒,但固定定时器的延迟有3秒,导致还未执行取样器,持续时间已经结束。

此时删掉固定定时器,运行结果

此时有人会有疑问。接口中设置的休眠时间就已经是3秒了,脚本中的持续时间还只是2秒,为什么这次成功执行了呢?

持续时间的设置,只作用于还未执行的取样器。已经执行的取样器,无论等待多长时间,都会执行完成。

本文示例接口源代码可从前言中下载。


Jmeter-线程组下篇的更多相关文章

  1. 【转】关于JMeter线程组中线程数,Ramp-Up Period,循环次数之间的设置概念

    关于JMeter线程组中线程数,Ramp-Up Period,循环次数之间的设置概念 笔者是个刚刚踏入压力测试领域不到2个月的小菜,这里分享一下线程组中3个参数之间关系的个人见解,不喜请!喷!,望大家 ...

  2. 关于JMeter线程组中线程数,Ramp-Up Period,循环次数之间的设置概念

    关于JMeter线程组中线程数,Ramp-Up Period,循环次数之间的设置概念 笔者是个刚刚踏入压力测试领域不到2个月的小菜,这里分享一下线程组中3个参数之间关系的个人见解,不喜请!喷!,望大家 ...

  3. JMeter线程组编辑区揭秘

    线程组编辑区如下: 有点复杂,但是慢慢看下来,还是比较容易理解. Name 带有业务含义的名字. Comments 线程组的备注说明. Action to be taken after a Sampl ...

  4. jmeter线程组之间传递参数

    JMeter 变量作用域局限于所属线程.这样设计是经过深思熟虑的,目的是让测试线程能够独立运转.有时候用户可能需要在不同线程间(可能属于同一个线程组,也可能不属于同一个线程组)传递变量. 其中一种方法 ...

  5. JMeter 线程组之ConcurrencyThreadGroup介绍

    线程组之ConcurrencyThreadGroup by:授客 QQ:1033553122 测试环境 apache-jmeter-3.2 jmeter-plugins-manager-1.3.jar ...

  6. JMeter 线程组之Stepping Thread Group插件介绍

    线程组之Stepping Thread Group插件介绍   by:授客 QQ:1033553122 测试环境 apache-jmeter-2.13   插件: https://jmeter-plu ...

  7. jmeter线程组之间传参

    背景介绍: 使用jmeter做登录和搜索接口的测试: 登录接口请求头为:Content-Type: application/x-www-form-urlencoded; charset=UTF-8 搜 ...

  8. jmeter线程组介绍

    Jmeter中的测试计划是一直有的,但可以在右侧修改名字,要开始做具体测试设计前,都需要在测试计划下边添加一个线程组,添加路径为鼠标捕获测试计划后,点击鼠标右键->添加->Threads( ...

  9. Jmeter线程组间传递参数

    现在做测试和以前不太一样了,以前只要站在一个用户的角度做端到端的UI测试就可以了,现在不会做接口测试,出去都不好意思和别人打招呼.那提到接口测试,就不得不提一下接口测试利器Jmeter,大家也都知道, ...

  10. jmeter线程组基本设置

    线程组基本设置 在线程组界面中可以设置以下数据,进行控制线程组: 1.取样器错误后要执行的动作: 继续:忽略错误,继续执行 Start Next Thread Loop: 忽略错误,线程当前循环终止, ...

随机推荐

  1. SQL优化篇之-如何减少耗时查询的调用次数

    函数调用次数与性能 在查询语句中,如果 Select 子句调用了较为耗时的函数或子查询,需要特别考虑函数调用次数对于SQL整体执行时间的影响. 一.数据准备,SQL 语句 模拟较耗时的用户函数 确保执 ...

  2. #分治,Dijkstra#洛谷 3350 [ZJOI2016]旅行者

    题目 给定一张\(n*m\)的网格图,\(q\)次询问两点之间距离 \(n*m\leq 2*10^4,q\leq 10^5\) 分析 首先floyd会TLE,考虑两点间距离可以由两段拼凑起来, 那么枚 ...

  3. #莫队,bitset#洛谷 3674 小清新人渣的本愿

    题目 分析 只要做到\(O(n\sqrt{n})\)的时间复杂度就可以了 考虑莫队,首先乘号就是枚举\(x\)的约数\(d\), 判断\(d\)和\(\frac{x}{d}\)是否同时出现, 再考虑差 ...

  4. ChatGPT商用网站源码+支持ai绘画(Midjourney)+GPT4.0+GPT3.5key绘画+Prompt角色+实时语音识别输入+后台一键版本更新!

    ChatGPT商用网站源码+支持ai绘画(Midjourney)+GPT4.0+GPT3.5key绘画+Prompt角色+实时语音识别输入+后台一键版本更新! 1.网站系统源码介绍: 程序已支持Cha ...

  5. 使用OHOS SDK构建libxml2

    参照OHOS IDE和SDK的安装方法配置好开发环境. 从github下载源码. 执行如下命令: git clone --depth=1 https://gitlab.gnome.org/GNOME/ ...

  6. 搜索引擎优化指南:SEO关键字、长尾关键字、短尾关键字以及反向链接

    内容 SEO SEO 代表"搜索引擎优化".它是一种数字营销策略,旨在提高网站或网页在搜索引擎未付费结果中的在线可见性.通常,网站在搜索结果页面中排名越高,或在搜索结果列表中显示的 ...

  7. Go 语言基础:包、函数、语句和注释解析

    一个 Go 文件包含以下几个部分: 包声明 导入包 函数 语句和表达式 看下面的代码,更好地理解它: 例子 package main import "fmt" func main( ...

  8. Numpy数组索引和切片

    数组可以通过索引或切片的方式进行访问或修改,数组切片x[start:stop:step],与Ptyhon内置的list标准索引和切片类似,只是数组产生的是一个非副本视图,根据条件索引的值如果修改,直接 ...

  9. 将Map中对应的key和value赋值到对象中

    BeanUtils位于import org.apache.commons.beanutils.BeanUtils包下 其使用方法: Map<String, Object> objectMa ...

  10. Spring源码 19 IOC getBean方法

    前面实现了 ClassPathXmlApplicationContext 的构造,接下来分析其调用的 getBean 方法. 以 getBean(UserDao.class) 为例. 1 Abstra ...