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即可继续

相关推荐
Mahir0817 小时前
Spring 循环依赖深度解密:从问题本质到三级缓存源码级解析
java·后端·spring·缓存·面试·循环依赖·三级缓存
RyFit18 小时前
SpringAI 常见问题及解决方案大全
java·ai
石山代码19 小时前
C++ 内存分区 堆区
java·开发语言·c++
绝知此事19 小时前
【算法突围 01】线性结构与哈希表:后端开发的收纳术
java·数据结构·算法·面试·jdk·散列表
无风听海19 小时前
C# 隐式转换深度解析
java·开发语言·c#
一只大袋鼠20 小时前
Git 进阶(二):分支管理、暂存栈、远程仓库与多人协作
java·开发语言·git
德思特21 小时前
从 Dify 配置页理解 RAG 的重要参数
java·人工智能·llm·dify·rag
YOU OU21 小时前
Spring IoC&DI
java·数据库·spring
один but you21 小时前
从可变参数到 emplace:现代 C++ 性能优化的核心组合
java·开发语言
是码龙不是码农1 天前
ThreadPoolExecutor 7 个核心参数详解
java·线程池·threadpool