性能工具之 MySQL OLTP Sysbench BenchMark 测试示例

举报
zuozewei 发表于 2024/08/03 11:02:34 2024/08/03
【摘要】 做为一名性能工程师掌握对 MySQL 的性能测试是非常必要的,本文基于 Sysbench 对MySQL OLTP(联机事务处理) 的 BenchMark 测试案例详细介绍具体方法。

一、前言

做为一名性能工程师掌握对 MySQL 的性能测试是非常必要的,本文基于 Sysbench 对MySQL OLTP(联机事务处理) 的 BenchMark 测试案例详细介绍具体方法。

二、测试环境

1、服务器配置

数据库服务器:

  • 操作系统:CentOS 7.6 64位
  • CPU:8核
  • 内存:16GB
  • 磁盘:500GB,最大吞吐量150 MB/s
  • 数据库版本:MySQL Community Server 8.0.37
  • 网络:局域网

测试服务器:

  • 操作系统:CentOS 7.6 64位
  • CPU:8核
  • 内存:16GB
  • 磁盘:500GB,最大吞吐量150 MB/s
  • 测试软件:sysbench-1.0.12
  • 网络:局域网

2、测试拓扑

在这里插入图片描述
📢注意:

  • 尽量不要在 MySQL 本服务器上进行测试,一方面可能无法体现网络(哪怕是局域网)的影响,另一方面,sysbench 的运行(并发数较高时)会影响挤压 MySQL 服务器性能。
  • 在开始 MySQL 测试之前,应针对数据库服务器做好 BenchMark 测试。

三、测试工具安装

Sysbench是一款基于LuaJIT的,模块化多线程基准测试工具,常用于数据库基准测试。通过内置的数据库测试模型,采用多线程并发操作来评估数据库的性能。了解Sysbench更多详情,请访问:https://github.com/akopytov/sysbench

本次测试使用的Sysbench版本为1.0.12,具体的安装命令如下:

# wget -c https://github.com/akopytov/sysbench/archive/1.0.12.zip
# yum install autoconf libtool mysql mysql-devel vim unzip
# unzip 1.0.12.zip
# cd sysbench-1.0.12
# ./autogen.sh
# ./configure
# make
# make install
#sysbench --version

显示以下内容说明已安装成功。
在这里插入图片描述

四、测试步骤

请根据实际信息,替换数据库、连接IP与用户密码。

1、导入数据

(1)使用 MySQL 命令或第三方工具登录数据库,并创建测试数据库 “loadtest” 。

mysql -u root -P 3306 -h -p -e "create database loadtest"

(2)使用 sysbench 命令导入测试背景数据到 “loadtest” 数据库。

sysbench
--test=/usr/local/share/sysbench/tests/include/oltp_legacy/oltp.lua
--db-driver=mysql --mysql-db=loadtest --mysql-user=root
--mysql-password= --mysql-port=3306 --mysql-host= --oltp-tables-count=64 --oltp-table-size=10000000 --num-threads=20 prepare

脚本参数及其含义:

  • –test:指定要运行的测试脚本,这里选择的是一个OLTP(在线事务处理)负载测试脚本。oltp.lua是一个预定义的脚本,用于模拟常见的数据库操作。
  • –db-driver:指定数据库驱动程序,这里选择的是 MySQL。
  • –mysql-db:指定要测试的 MySQL 数据库名称,这里是loadtest数据库。
  • –mysql-user:指定用于连接 MySQL 数据库的用户名,这里是 root 用户。
  • mysql-password:指定用于连接 MySQL 数据库的密码,这里为空,意味着没有设置密码(不推荐在生产环境中使用空密码)。
  • –mysql-port:指定 MySQL 服务器监听的端口,这里是默认的 3306 端口。
  • –mysql-host:指定 MySQL 服务器的主机地址,这里为空,表示连接本地数据库。
  • –oltp-tables-count:指定用于测试的表的数量,这里是 64 个表。
  • –oltp-table-size:指定每个表中的行数,这里是 10,000,000 行。表示每个表有一千万条记录。
  • –num-threads:指定测试时使用的线程数,这里是 20 个线程。表示并发 20 个线程进行测试。
  • prepare:测试提前准备数据

本文是生成 64 张表,每张表有1千万数据,合计导入6亿4千万条数据。

显示下面信息说明已经成功完成测试数据生成:

WARNING: the --test option is deprecated. You can pass a script name or path on the command line without any options.
WARNING: --num-threads is deprecated, use --threads instead
sysbench 1.0.12 (using bundled LuaJIT 2.1.0-beta2)

......
Inserting 10000000 records into 'sbtest63'
Creating secondary indexes on 'sbtest63'...
Creating table 'sbtest64'...
Inserting 10000000 records into 'sbtest64'
Creating secondary indexes on 'sbtest64'...
[root@ecs-825d-1113052 ~]#

生产的表结构如下:

CREATE TABLE sbtest (
id INTEGER UNSIGNED NOT NULL AUTO_INCREMENT,
k INTEGER UNSIGNED DEFAULT '0' NOT NULL,
c CHAR(120) DEFAULT '' NOT NULL,
pad CHAR(60) DEFAULT '' NOT NULL,
PRIMARY KEY (id)
) ENGINE=InnoDB

生产数据样例如下:
在这里插入图片描述

这里用到 oltp.lua 这个关键脚本,我们单独拿出分析下,源码如下:

[root@ecs-825d-1113052 ~]# cat /usr/local/share/sysbench/tests/include/oltp_legacy/oltp.lua
-- 匹配test路径并检查
pathtest = string.match(test, "(.*/)")
if pathtest then
   dofile(pathtest .. "common.lua")
else
   require("common")
end

-- 线程初始化函数
function thread_init()
   -- 设置变量
   set_vars()

   -- 检查数据库驱动和表引擎类型
   if (((db_driver == "mysql") or (db_driver == "attachsql")) and mysql_table_engine == "myisam") then
      local i
      local tables = {}
      -- 为每个表构建锁定语句
      for i=1, oltp_tables_count do
         tables[i] = string.format("sbtest%i WRITE", i)
      end
      -- 设置锁定和解锁查询
      begin_query = "LOCK TABLES " .. table.concat(tables, " ,")
      commit_query = "UNLOCK TABLES"
   else
      -- 默认使用事务的开始和提交语句
      begin_query = "BEGIN"
      commit_query = "COMMIT"
   end
end

-- 获取范围查询的条件字符串
function get_range_str()
   local start = sb_rand(1, oltp_table_size)
   return string.format(" WHERE id BETWEEN %u AND %u",
                        start, start + oltp_range_size - 1)
end

-- 定义事件函数
function event()
   local rs
   local i
   local table_name
   local c_val
   local pad_val
   local query

   -- 随机选择一个表
   table_name = "sbtest".. sb_rand_uniform(1, oltp_tables_count)
   -- 如果没有跳过事务,则开始事务
   if not oltp_skip_trx then
      db_query(begin_query)
   end

   -- 如果不是仅写操作
   if not oltp_write_only then
      -- 执行点查询
      for i=1, oltp_point_selects do
         rs = db_query("SELECT c FROM ".. table_name .." WHERE id=" .. sb_rand(1, oltp_table_size))
      end

      -- 如果需要执行范围查询
      if oltp_range_selects then
         -- 简单范围查询
         for i=1, oltp_simple_ranges do
            rs = db_query("SELECT c FROM ".. table_name .. get_range_str())
         end

         -- 范围求和查询
         for i=1, oltp_sum_ranges do
            rs = db_query("SELECT SUM(K) FROM ".. table_name .. get_range_str())
         end

         -- 范围排序查询
         for i=1, oltp_order_ranges do
            rs = db_query("SELECT c FROM ".. table_name .. get_range_str() .. " ORDER BY c")
         end

         -- 范围去重查询
         for i=1, oltp_distinct_ranges do
            rs = db_query("SELECT DISTINCT c FROM ".. table_name .. get_range_str() .. " ORDER BY c")
         end
      end
   end

   -- 如果不是只读操作
   if not oltp_read_only then
      -- 执行索引更新
      for i=1, oltp_index_updates do
         rs = db_query("UPDATE " .. table_name .. " SET k=k+1 WHERE id=" .. sb_rand(1, oltp_table_size))
      end

      -- 执行非索引更新
      for i=1, oltp_non_index_updates do
         c_val = sb_rand_str("###########-###########-###########-###########-###########-###########-###########-###########-###########-###########")
         query = "UPDATE " .. table_name .. " SET c='" .. c_val .. "' WHERE id=" .. sb_rand(1, oltp_table_size)
         rs = db_query(query)
         if rs then
            print(query)
         end
      end

      -- 执行删除和插入操作
      for i=1, oltp_delete_inserts do
         i = sb_rand(1, oltp_table_size)
         rs = db_query("DELETE FROM " .. table_name .. " WHERE id=" .. i)
         c_val = sb_rand_str("###########-###########-###########-###########-###########-###########-###########-###########-###########-###########")
         pad_val = sb_rand_str("###########-###########-###########-###########-###########")
         rs = db_query("INSERT INTO " .. table_name ..  " (id, k, c, pad) VALUES " .. string.format("(%d, %d, '%s', '%s')",i, sb_rand(1, oltp_table_size) , c_val, pad_val))
      end
   end

   -- 如果没有跳过事务,则提交事务
   if not oltp_skip_trx then
      db_query(commit_query)
   end
end

这段 oltp.lua 代码的主要步骤如下:

  • 路径匹配与加载配置:
    • 检查并获取脚本的路径。
    • 如果路径存在,加载common.lua 文件;否则使用 require 函数加载模块。
  • 线程初始化 (thread_init):
    • 初始化变量。
    • 根据数据库驱动和表引擎类型,决定是否使用锁表操作。
    • 如果数据库驱动是 mysql 或 attachsql 且表引擎为 myisam,则构建锁定和解锁查询语句。
    • 否则,使用默认的事务控制语句(BEGIN 和 COMMIT)。
  • 获取范围查询字符串 (get_range_str):
    • 随机生成一个起始ID。
    • 返回一个用于范围查询的条件字符串,指定查询范围为从起始ID到起始ID加上范围大小减去1。
  • 事件处理 (event):
    • 定义事件函数,该函数是 Sysbench 测试的核心部分
    • 事件函数包括以下操作:
      • 随机选择一个表。
      • 如果没有跳过事务,则开始事务。
      • 根据配置执行不同类型的查询和更新操作,包括点查询、范围查询、索引更新、非索引更新、删除和插入操作。
      • 范围查询包括简单范围查询、求和范围查询、排序范围查询和去重范围查询。
      • 如果没有跳过事务,则提交事务。

这段代码是典型的OLTP(联机事务处理)负载测试脚本,通过模拟多种数据库操作(查询、更新、删除、插入),来评估数据库在高并发访问场景下的性能表现。

2、压测数据

sysbench
--test=/usr/local/share/sysbench/tests/include/oltp_legacy/oltp.lua
--db-driver=mysql --mysql-db=loadtest --mysql-user=root
--mysql-password= --mysql-port=3306 --mysql-host=--oltp-tables-count=64
--oltp-table-size=10000000 --max-time=3600 --max-requests=0
--num-threads=200 --report-interval=3 --forced-shutdown=1 run

脚本参数及其含义:

  • –test:指定要运行的测试脚本,这里选择的是一个OLTP(在线事务处理)负载测试脚本。oltp.lua是一个预定义的脚本,用于模拟常见的数据库操作。
  • –db-driver:指定数据库驱动程序,这里选择的是 MySQL。
  • –mysql-db:指定要测试的 MySQL 数据库名称,这里是 loadtest 数据库。
  • –mysql-user:指定用于连接 MySQL 数据库的用户名,这里是 root 用户。
  • mysql-password:指定用于连接 MySQL 数据库的密码,这里为空,意味着没有设置密码(不推荐在生产环境中使用空密码)。
  • –mysql-port:指定 MySQL 服务器监听的端口,这里是默认的 3306 端口。
  • –mysql-host:指定 MySQL 服务器的主机地址,这里为空,表示连接本地数据库。
  • –oltp-tables-count:指定用于测试的表的数量,这里是 64 个表。
  • –oltp-table-size:指定每个表中的行数,这里是 10,000,000 行。表示每个表有一千万条记录。
  • –max-time:指定测试的最大持续时间为3600秒(1小时)。
  • –max-requests:指定要执行的最大请求数。值为0表示请求数不受限制,直到达到最大时间。
  • –num-threads:指定测试时使用的线程数,这里是 200 个线程。表示并发 200 个线程进行测试。
  • –report-interval:指定报告中间结果的时间间隔(每3秒报告一次)。
  • –forced-shutdown:指定如果达到最大时间,Sysbench应该强制关闭测试(1表示启用)。
  • run:开始运行测试的命令。

简要说明就是并发200线程,压测1小时,每3秒打印一次结果等。

3、清理数据

测试完成后,可以运行以下脚本清理测试数据:

sysbench
--test=/usr/local/share/sysbench/tests/include/oltp_legacy/oltp.lua
--db-driver=mysql --mysql-db=loadtest --mysql-user=root
--mysql-password= --mysql-port=3306 --mysql-host= --oltp-tables-count=64 --oltp-table-size=10000000--max-time=3600 --max-requests=0 --num-threads=200 cleanup

脚本参数及其含义:

  • –test:指定要运行的测试脚本,这里选择的是一个OLTP(在线事务处理)负载测试脚本。oltp.lua是一个预定义的脚本,用于模拟常见的数据库操作。
  • –db-driver:指定数据库驱动程序,这里选择的是 MySQL。
  • –mysql-db:指定要测试的 MySQL 数据库名称,这里是 loadtest 数据库。
  • –mysql-user:指定用于连接 MySQL 数据库的用户名,这里是 root 用户。
  • mysql-password:指定用于连接 MySQL 数据库的密码,这里为空,意味着没有设置密码(不推荐在生产环境中使用空密码)。
  • –mysql-port:指定 MySQL 服务器监听的端口,这里是默认的 3306 端口。
  • –mysql-host:指定 MySQL 服务器的主机地址,这里为空,表示连接本地数据库。
  • –oltp-tables-count:指定用于测试的表的数量,这里是 64 个表。
  • –oltp-table-size:指定每个表中的行数,这里是 10,000,000 行。表示每个表有一千万条记录。
  • –max-time:指定测试的最大持续时间为3600秒(1小时)。
  • –max-requests:指定要执行的最大请求数。值为0表示请求数不受限制,直到达到最大时间。
  • –num-threads:指定测试时使用的线程数,这里是 200 个线程。表示并发 200 个线程进行测试。
  • cleanup:测试完成后对数据库进行清理。

五、结果解析

以下为压测过程中打印的结果:

[ 3522s ] thds: 200 tps: 153.98 qps: 3119.87 (r/w/o: 2155.68/656.24/307.95) lat (ms,95%): 1235.62 err/s: 0.00 reconn/s: 0.00
[ 3525s ] thds: 200 tps: 157.36 qps: 2992.89 (r/w/o: 1997.37/680.79/314.72) lat (ms,95%): 4358.09 err/s: 0.00 reconn/s: 0.00
[ 3528s ] thds: 200 tps: 85.33 qps: 1852.86 (r/w/o: 1400.23/281.98/170.65) lat (ms,95%): 1258.08 err/s: 0.00 reconn/s: 0.00

测试结束后,查看输出文件,如下所示:

FATAL: The --max-time limit has expired, forcing shutdown...
SQL statistics:
    queries performed:
        read:                            5358024
        write:                           1530377
        other:                           765297
        total:                           7653698
    transactions:                        382581 (106.24 per sec.)
    queries:                             7653698 (2125.42 per sec.)
    ignored errors:                      0      (0.00 per sec.)
    reconnects:                          0      (0.00 per sec.)

Number of unfinished transactions on forced shutdown: 200

General statistics:
    total time:                          3601.0196s
    total number of events:              382581

Latency (ms):
         min:                                  4.72
         avg:                               1881.83
         max:                              10972.92
         95th percentile:                   4128.91
         sum:                            719951371.94

Threads fairness:
    events (avg/stddev):           1913.9050/24.88
    execution time (avg/stddev):   3599.7569/1.81

是不是有点晕,那我们稍微翻译下,如下所示:

FATAL: The --max-time limit has expired, forcing shutdown...
#SQL统计部分表明了总查询量以及每秒执行的查询和事务数量。这些数据有助于了解数据库的处理能力和性能表现。
SQL statistics(SQL统计信息):
    queries performed(查询执行情况):
        read(读查询):                          5358024
        write(写查询):                         1530377
        other(其它查询):                        765297
        total(总查询):                         7653698
    transactionss(事务):                        382581 (106.24 per sec.) (每秒106.24次)
    queries(查询):                             7653698 (2125.42 per sec.) (每秒2125.42次)
    ignored errors(忽略的错误):                      0      (0.00 per sec.) (每秒0次)
    reconnect(重连)s:                          0      (0.00 per sec.) (每秒0次)
    
# 强制关闭时未完成的事务数量为200,表明在测试过程中有200个事务未能完成,这可能与测试环境或配置有关。
Number of unfinished transactions on forced shutdown: 200

General statistics(一般统计信息):
    total time(总时间):                          3601.0196s
    total number of events(事件总数):              382581
    
#延迟数据展示了不同百分位的延迟情况,这些数据对分析数据库响应时间和性能瓶颈很有用。
Latency(延迟) (ms):
         min(最小延迟):                                  4.72
         avg(平均延迟):                               1881.83
         max(最大延迟):                              10972.92
         95th percentile(95%分位延迟):                4128.91
         sum(延迟总和):                          719951371.94
         
#线程公平性数据表明,每个线程处理的事件数的平均值和标准差,以及每个线程的执行时间的平均值和标准差。
Threads fairness(线程公平性):
    events (avg/stddev)(事件(平均值/标准差)):           1913.9050/24.88
    execution time (avg/stddev)(执行时间(平均值/标准差)):   3599.7569/1.81

这些数据展示了MySQL在高并发负载下的性能情况,主要关注点包括:

  • 查询和事务的执行率:每秒查询和事务数量表明了数据库的吞吐量。
  • 延迟:延迟数据(平均、最大和95%分位)显示了数据库的响应时间和性能瓶颈。
  • 未完成事务:强制关闭时未完成的事务数提示了潜在的事务处理问题。
  • 线程公平性:线程间的负载均衡情况,标准差较低表示负载分配较为均衡。

主要关注的性能指标有:

  • TPS :Transaction Per Second,数据库每秒执行的事务数,每个事务中包含18条SQL语句。
  • QPS :Query Per Second,数据库每秒执行的SQL数,包含insert、select、update、delete等。
  • 延迟:Latency,数据库执行的事务耗时。

Sysbench默认提交的事务中包含18条SQL语句,具体执行语句和条数如下:

主键SELECT语句,10条:
SELECT c FROM ​{rand_table_name} where id={rand_id};

范围SELECT语句,4条:
SELECT c FROM ​{rand_table_name} WHERE id BETWEEN {rand_id_start} AND ${rand_id_end};
SELECT SUM(K) FROM ​{rand_table_name} WHERE id BETWEEN {rand_id_start} AND ${rand_id_end};
SELECT c FROM ​{rand_table_name} WHERE id BETWEEN {rand_id_start} AND ${rand_id_end} ORDER BY c;
SELECT DISTINCT c FROM ​{rand_table_name} WHERE id BETWEEN {rand_id_start} AND ${rand_id_end} ORDER BY c;

UPDATE语句,2条:
UPDATE ​{rand_table_name} SET k=k+1 WHERE id={rand_id}
UPDATE ​{rand_table_name} SET c={rand_str} WHERE id=${rand_id}

DELETE语句,1条:
DELETE FROM ​{rand_table_name} WHERE id={rand_id}

INSERT语句,1条:
INSERT INTO ​{rand_table_name} (id, k, c, pad) VALUES ({rand_id},​{rand_k},{rand_str_c},${rand_str_pad})

这些结果可以用于性能调优和瓶颈分析,从而提升 MySQL 数据库在实际应用中的表现。

从Sysbench测试结果来看,这台MySQL服务器在高并发负载下的性能表现有以下几个关键点:

  • 事务处理能力:
    • 每秒事务数(TPS)为106.24次。
    • 总事务数为 382581。
  • 查询处理能力:
    • 每秒查询数(QPS)为 2125.42次。
    • 总查询数为 7653698。
  • 延迟:
    • 平均延迟为 1881.83 毫秒,较高,说明在负载压力下,响应时间比较长
    • 最大延迟为 10972.92 毫秒,非常高,表明在高负载下可能存在严重的性能瓶颈
    • 95% 分位延迟为 4128.91 毫秒,表示大多数请求的响应时间在 4 秒以上,体验较差
  • 未完成事务:
    • 强制关闭时未完成的事务数为 200,表明在高负载下有一部分事务未能及时处理完成。
  • 线程公平性:
    • 每个线程处理的事件数的标准差为 24.88,表明线程间的负载分配较为均衡。
    • 每个线程的执行时间的标准差为 1.81,表明线程执行时间也较为一致。

这里我们可以对比下某云的测试结果:
在这里插入图片描述

数据服务器资源监控数据:
在这里插入图片描述
我们可以看到CPU峰值到75%左右,磁盘峰值写入速率达50MB/s,峰值读取速率达 100MB/s。

六、最后

我们可以看到从测试结果的结果来看,MySQL数据库的性能表现并不好,那么我们接下来应对MySQL数据库进行性能调优并再次验证,希望本文能对你的工作带来一点点帮助,如果有用别忘了点个赞,多谢。

【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。