Gatling压测案例

引言:公司某个网站需要做压测,验证视频分片功能承载能力,刚开始选择了Apache Jmeter,发现Jmeter占用资源太大了,在虚拟机跑500个线程并发就内存溢出了。因为Jmeter一条线程就是一条用户线程,每条线程都需要占用一定空间的内存,因此在高并发量下需要大量资源。一个资深同事建议试试Gatling,它是异步非阻塞线程,对资源占用小且速度个人感觉比Jmeter快,缺点是没有GUI,需要自己写脚本

Windows测试

直接在官网下一个gatling-maven-plugin-demo-java-main.zip,里面有一个可以直接运行的demo

java 复制代码
package example;

import static io.gatling.javaapi.core.CoreDsl.*;
import static io.gatling.javaapi.http.HttpDsl.*;

import io.gatling.javaapi.core.*;
import io.gatling.javaapi.http.*;

public class BasicSimulation extends Simulation {

  // Load VU count from system properties
  // Reference: https://docs.gatling.io/guides/passing-parameters/
  private static final int vu = Integer.getInteger("vu", 1);

  // Define HTTP configuration
  // Reference: https://docs.gatling.io/reference/script/protocols/http/protocol/
  private static final HttpProtocolBuilder httpProtocol = http.baseUrl("https://api-ecomm.gatling.io")
      .acceptHeader("application/json")
      .userAgentHeader(
          "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/134.0.0.0 Safari/537.36");

  // Define scenario
  // Reference: https://docs.gatling.io/reference/script/core/scenario/
  private static final ScenarioBuilder scenario = scenario("Scenario").exec(http("Session").get("/session"));

  // Define assertions
  // Reference: https://docs.gatling.io/reference/script/core/assertions/
  private static final Assertion assertion = global().failedRequests().count().lt(1L);

  // Define injection profile and execute the test
  // Reference: https://docs.gatling.io/reference/script/core/injection/
  {
    setUp(scenario.injectOpen(atOnceUsers(vu))).assertions(assertion).protocols(httpProtocol);
  }
}

直接在terminal执行命令./mvnw clean install ./mvnw gatling:test即可

如果是在cmd里面需要更改一下命令mvnw.cmd gatling:test

Demo里面定义了默认并发数量是1
private static final int vu = Integer.getInteger("vu", 1);

可以直接改成自己想要的并发数,也可以通过options指定

如500并发数:./mvnw gatling:test -Dvu=500

Linux测试

在Linux测试最方便就是用docker跑。

1.先准备好.scale测试脚本

(上面的demo是java写的,现在需要一个scale写的脚本)

例如我自己写的关于视频切片功能测试的脚本:

scale 复制代码
package example

import io.gatling.core.Predef._
import io.gatling.http.Predef._
import scala.concurrent.duration.Duration.Inf
import scala.util.Random

class BasicSimulation extends Simulation {

  // 视频大小 (例如 800MB) ,不要超过目标视频实际大小
  val VIDEO_SIZE_BYTES = 800L * 1024 * 1024

  // 模拟用户每次拖动加载的数据块大小 (例如 3MB)
  val CHUNK_SIZE_BYTES = 3L * 1024 * 1024

  // 获取虚拟用户数量,优先读取系统属性 -Dvu=xx,否则默认为 1
  val vu = Option(System.getProperty("vu")).map(_.toInt).getOrElse(1)

  // 1. 定义 HTTP 协议配置
  val httpProtocol = http
    .baseUrl("http://10.211.2.123:8080")
    .userAgentHeader("Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/123.0.0.0 Safari/537.36")
    .acceptHeader("*/*")
    .connectionHeader("keep-alive")

  // 2. 定义场景
  val myScenario = scenario("Video Range Request")
    .exec { session =>
      // 生成 0 到 (视频大小 - 块大小) 之间的随机数
      val start = Math.abs(Random.nextLong()) % (VIDEO_SIZE_BYTES - CHUNK_SIZE_BYTES)
      val end = start + CHUNK_SIZE_BYTES - 1
      val rangeValue = s"bytes=$start-$end"
      println(s"Range: $rangeValue")
      session.set("rangeHeader", rangeValue)
    }
    .exec(
      http("Get Random Video Chunk")
        // 视频path
        .get("/test.mp4")
        .header("Range", "${rangeHeader}")
        .header("Referer", "http://10.211.2.123:8080/test.mp4")
        .check(
          bodyBytes.transform { (bytes: Array[Byte]) =>
            println(s"Received ${bytes.length} bytes")
            bytes
          }
        )
        .check(status.in(200, 206))
    )

  // 3. assertion
  val assertion = global.failedRequests.count.lt(1)

  // 4. start
  setUp(
    myScenario.inject(
      atOnceUsers(vu)
    )
  ).protocols(httpProtocol)
   .assertions(assertion)
}

2.准备好文件目录

在 Linux 服务器上,创建一个文件夹(比如叫 my-test),并按照 Gatling 的标准结构放好文件:

复制代码
my-test/
├── results/                # (空文件夹,用来放报告)
└── user-files/             # 关键目录
    └── simulations/        # 把你的scala script放这里
        └── BasicSimulation.scala

3.开启容器测试

进入 my-test 目录,run command

bash 复制代码
docker run -it --rm \
  -e JAVA_OPTS="-Dvu=10" \
  -v $(pwd)/results:/opt/gatling/results \
  -v $(pwd)/user-files:/opt/gatling/user-files \
  denvazh/gatling
  • -v $(pwd)/user-files:/opt/gatling/user-files

    这句话的意思是:"把服务器上当前目录下的 user-files 文件夹,替换掉容器里原本的 user-files 文件夹"。

  • 这样,容器启动后,就能直接看到你上传的 .scala 脚本了。

  • JAVA_OPTS="-Dvu=10" 指定concurrent数量

如出现Select run description (optional),enter即可继续

相关推荐
小旭95279 分钟前
Spring Security 实现权限控制(认证 + 授权全流程)
java·后端·spring
金銀銅鐵15 分钟前
[Java] 如何通过 cglib 的 FastClass 调用一个类中的“任意”方法?
java·后端
阿维的博客日记33 分钟前
为什么会增加TreeMap和TreeSet这两类,有什么核心优势吗?可以解决什么核心痛点?
java·treeset·treemap
dllxhcjla42 分钟前
黑马头条1
java
宠友信息1 小时前
一套基于uniapp+springboot完整社区系统是如何实现的?友猫社区源码级功能解析
java·spring boot·后端·微服务·微信·uni-app
humors2211 小时前
各厂商工具包网址
java·数据库·python·华为·sdk·苹果·工具包
无限进步_1 小时前
【C++&string】大数相乘算法详解:从字符串加法到乘法实现
java·开发语言·c++·git·算法·github·visual studio
海兰1 小时前
使用 Spring AI 打造企业级 RAG 知识库第二部分:AI 实战
java·人工智能·spring
历程里程碑2 小时前
二叉树---二叉树的中序遍历
java·大数据·开发语言·elasticsearch·链表·搜索引擎·lua
小信丶2 小时前
Spring Cloud Stream EnableBinding注解详解:定义、应用场景与示例代码
java·spring boot·后端·spring