性能工具之 MySQL OLTP Sysbench BenchMark 测试示例
文章目录
- 一、前言
- 二、测试环境
- 1、服务器配置
- 2、测试拓扑
- 三、测试工具安装
- 四、测试步骤
- 1、导入数据
- 2、压测数据
- 3、清理数据
- 五、结果解析
- 六、最后
一、前言
做为一名性能工程师掌握对 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 thendofile(pathtest .. "common.lua")
elserequire("common")
end-- 线程初始化函数
function thread_init()-- 设置变量set_vars()-- 检查数据库驱动和表引擎类型if (((db_driver == "mysql") or (db_driver == "attachsql")) and mysql_table_engine == "myisam") thenlocal ilocal tables = {}-- 为每个表构建锁定语句for i=1, oltp_tables_count dotables[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 rslocal ilocal table_namelocal c_vallocal pad_vallocal query-- 随机选择一个表table_name = "sbtest".. sb_rand_uniform(1, oltp_tables_count)-- 如果没有跳过事务,则开始事务if not oltp_skip_trx thendb_query(begin_query)end-- 如果不是仅写操作if not oltp_write_only then-- 执行点查询for i=1, oltp_point_selects dors = 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 dors = db_query("SELECT c FROM ".. table_name .. get_range_str())end-- 范围求和查询for i=1, oltp_sum_ranges dors = db_query("SELECT SUM(K) FROM ".. table_name .. get_range_str())end-- 范围排序查询for i=1, oltp_order_ranges dors = db_query("SELECT c FROM ".. table_name .. get_range_str() .. " ORDER BY c")end-- 范围去重查询for i=1, oltp_distinct_ranges dors = db_query("SELECT DISTINCT c FROM ".. table_name .. get_range_str() .. " ORDER BY c")endendend-- 如果不是只读操作if not oltp_read_only then-- 执行索引更新for i=1, oltp_index_updates dors = 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 doc_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 thenprint(query)endend-- 执行删除和插入操作for i=1, oltp_delete_inserts doi = 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))endend-- 如果没有跳过事务,则提交事务if not oltp_skip_trx thendb_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: 5358024write: 1530377other: 765297total: 7653698transactions: 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: 200General statistics:total time: 3601.0196stotal number of events: 382581Latency (ms):min: 4.72avg: 1881.83max: 10972.9295th percentile: 4128.91sum: 719951371.94Threads fairness:events (avg/stddev): 1913.9050/24.88execution time (avg/stddev): 3599.7569/1.81
是不是有点晕,那我们稍微翻译下,如下所示:
FATAL: The --max-time limit has expired, forcing shutdown...
#SQL统计部分表明了总查询量以及每秒执行的查询和事务数量。这些数据有助于了解数据库的处理能力和性能表现。
SQL statistics(SQL统计信息):queries performed(查询执行情况):read(读查询): 5358024write(写查询): 1530377other(其它查询): 765297total(总查询): 7653698transactionss(事务): 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: 200General statistics(一般统计信息):total time(总时间): 3601.0196stotal number of events(事件总数): 382581#延迟数据展示了不同百分位的延迟情况,这些数据对分析数据库响应时间和性能瓶颈很有用。
Latency(延迟) (ms):min(最小延迟): 4.72avg(平均延迟): 1881.83max(最大延迟): 10972.9295th percentile(95%分位延迟): 4128.91sum(延迟总和): 719951371.94#线程公平性数据表明,每个线程处理的事件数的平均值和标准差,以及每个线程的执行时间的平均值和标准差。
Threads fairness(线程公平性):events (avg/stddev)(事件(平均值/标准差)): 1913.9050/24.88execution 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数据库进行性能调优并再次验证,希望本文能对你的工作带来一点点帮助,如果有用别忘了点个赞,多谢。
相关文章:

性能工具之 MySQL OLTP Sysbench BenchMark 测试示例
文章目录 一、前言二、测试环境1、服务器配置2、测试拓扑 三、测试工具安装四、测试步骤1、导入数据2、压测数据3、清理数据 五、结果解析六、最后 一、前言 做为一名性能工程师掌握对 MySQL 的性能测试是非常必要的,本文基于 Sysbench 对MySQL OLTP(联…...

【QT】QCustomPlot库中iSelectPlottables的使用
QCP::iSelectPlottables 是 QCustomPlot 库中的一个枚举值,用于控制选择交互。QCustomPlot 是一个用于创建绘图和数据可视化的Qt库。 QCP::iSelectPlottables 允许用户选择图表中的绘图对象(如图形、曲线、柱状图等)。 应用场景 QCP::iSele…...

字节跳动联手博通:5nm AI芯片诞生了?
字节跳动联手博通:5nm AI芯片诞生了? 前言 就在6月24日,字节跳动正在与美国博通合作开发一款5纳米工艺的专用集成电路(ASIC) AI处理器。这款芯片旨在降低采购成本并确保高端AI芯片的稳定供应。 根据报道,尽管芯片设计工作进展顺利…...

【数据结构与算法】动态查找表(二叉排序树,二叉平衡树)详解
二叉排序树的数据结构。 struct TreeNode {ElemType data;TreeNode *left, *right; }; using BiTree TreeNode *;结构体包含三个成员: data 是一个 ElemType 类型的变量,用于存储二叉搜索树节点的数据。left 是一个指向 TreeNode 类型的指针ÿ…...

PyTorch中“No module named ‘torch._six‘“的报错场景及处理方法
🎬 鸽芷咕:个人主页 🔥 个人专栏: 《C干货基地》《粉丝福利》 ⛺️生活的理想,就是为了理想的生活! 引入 在使用PyTorch时,您可能会遇到"No module named ‘torch._six’"的错误。这通常是因为PyTorch的某些…...

Spring Boot 集成 MinIO 实现文件上传
Spring Boot 集成 MinIO 实现文件上传 一、 Minio 服务准备 MinIO的搭建过程参考 Docker 搭建 MinIO 对象存储。 登录MinIO控制台,新建一个 Bucket,修改 Bucket 权限为公开。 二、MinIO 集成 添加 MinIO 依赖 <!-- https://mvnrepository.com/ar…...

目标跟踪——KCF源码用python实现
from numpy.fft import fft2, ifft2, fftshift import cv2 import numpy as npclass HOG:def __init__(self, winSize):""":param winSize: 检测窗口的大小"""self.winSize winSizeself.blockSize (8, 8)self.blockStride (4, 4)self.cellSiz…...

前端 转换笔记
<!DOCTYPE html> <html> <head> <meta charset"utf-8" /> <title>转换</title> <style> .box{ /* 盒子摆在body的正中间 */ position: absolut…...

个人开发笔记
开发笔记 开发常见问题Vue开发中页面flex滚动布局,内容置顶问题功能快捷键 开发常见问题 Vue开发中页面flex滚动布局,内容置顶问题 直接操作路由: const router createRouter({routes: routes,history: createWebHashHistory(),scrollBeha…...

pdf压缩,pdf压缩在线,pdf文件太大怎么变小
在数字化时代,PDF文档因其跨平台、保持原样、易于阅读和打印等特点,成为了我们日常工作和生活中不可或缺的一部分。然而,随着PDF文件的不断累积,存储空间逐渐变得紧张,特别是在处理大量大型PDF文件时,如何有…...

Go 如何使用指针灵活操作内存
💝💝💝欢迎莅临我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 推荐:「stormsha的主页」…...

【面试干货】Java中的++操作符与线程安全性
【面试干货】Java中的操作符与线程安全性 1、什么是线程安全性?2、 操作符的工作原理3、 操作符与线程安全性4、如何确保线程安全?5、 结论 💖The Begin💖点点关注,收藏不迷路💖 在Java编程中,操…...

NLP学习与踩坑记录(持续更新版)
NLP学习与踩坑记录(持续更新版) OSError: Cant load tokenizer for bert-base-uncased.google.protobuf.message.DecodeError: Error parsing messageDeepspeed 本博客记录了博主在学习NLP时遇到了各种各样的问题与解决方法,供大家参考&#…...

Java也能做OCR!SpringBoot 整合 Tess4J 实现图片文字识别
文章目录 1. 环境准备1.1 安装 Tesseract OCR 引擎1.2 引入 Tess4J 依赖 2. 创建 Spring Boot 项目2.1 初始化项目2.2 目录结构 3. 编写 OCR 功能代码3.1 创建服务层3.2 创建控制器层 4. 配置 Tesseract 语言包5. 运行和测试5.1 启动 Spring Boot 应用5.2 使用 Postman 或 cURL…...

微信小程序常用标签及其用法
大家好,我是linzi,今天我来给大家分享一下微信小程序一些个常用的标签及其用法 1. <view> 标签 <view> 标签是小程序中最常用的标签之一,用于组织和布局页面上的内容,类似于HTML中的 <div> 标签。 <view …...

开发查询订单信息fastGPT智能体工作流 将工作流接入到人工客服系统
我在抖音上发布了视频 https://www.douyin.com/video/7382446337482099977 下面是主要内容介绍 【视频标题:】开发查询订单信息fastGPT智能体工作流 将工作流接入到人工客服系统 #智能体 #FastGPT #客服系统-----------【视频行业分类:】<3C数码>-…...

Flink集群运行模式
我们了解了flink的一个集群的一个基础架构,包括里面核心的一些组件,比如说job manager,task manager等一些组件的一些主要的一些组成。本节课程开始我们学习flink的一个集群部署模式。首先我们来看一下flink集群部署模式究竟应该有哪一些种类…...

XSS 安全漏洞介绍及修复方案
简介 XSS(Cross Site Scripting)是一种常见的 Web 安全漏洞,攻击者通过在网页中注入恶意脚本代码,使得网页在用户端执行这些脚本,从而窃取用户信息或者进行其他恶意操作。为了防止 XSS 攻击,可以使用正则表…...

基于STM32的智能仓库管理系统
目录 引言环境准备智能仓库管理系统基础代码实现:实现智能仓库管理系统 4.1 数据采集模块4.2 数据处理与分析4.3 通信模块实现4.4 用户界面与数据可视化应用场景:仓库管理与优化问题解决方案与优化收尾与总结 1. 引言 智能仓库管理系统通过使用STM32嵌…...

LeetCode —— 只出现一次的数字
只出现一次的数字 I 本题依靠异或运算符的特性,两个相同数据异或等于0,数字与0异或为本身即可解答。代码如下: class Solution { public:int singleNumber(vector<int>& nums) {int ret 0;for (auto e : nums){ret ^ e;}return ret;} };只出…...

python遍历文件夹中所有图片
python遍历文件夹中的图片-CSDN博客 这个是之前的版本,现在这个版本会更好,直接进来就在列表中 path glob.glob("1/*.jpg")print(path)print(len(path))path_img glob.glob("1/*.jpg")path_img.extend(path)print(len(path_img))…...

速盾:DDOS能打死高防ip吗?
DDoS攻击是一种利用大量计算机或设备发起的分布式拒绝服务攻击。它的目标是通过发送大量流量或请求,使目标服务器或网络资源无法正常工作。高防IP是一种具有强大防御能力的网络服务,能够抵御各种形式的网络攻击,包括DDoS攻击。然而࿰…...

3dsMax怎样让渲染效果更逼真出色?三套低中高参数设置
渲染是将精心构建的3D模型转化为逼真图像的关键步骤。但要获得令人惊叹的渲染效果,仅仅依赖默认设置是不够的。 实现在追求极致画面效果的同时,兼顾渲染速度和时间还需要进行一些调节设置,如何让渲染效果更加逼真? 一、全局照明与…...

Android的OverlayFS原理与作用
标签: OverlayFS; Android;Overlay Filesystem; Android的OverlayFS原理与作用 概述 OverlayFS(Overlay Filesystem)是一种联合文件系统,允许将一个或多个文件系统叠加在一起,使它们表现为一个单一的文件系统。Android系统利用OverlayFS来实现动态文件系统的叠加和管…...

奇点临近:人类与智能时代的未来
在信息爆炸的时代,我们每天都被海量的信息所淹没,如何才能在这个嘈杂的世界中找到真正有价值的信息?如何才能利用信息的力量,提升我们的认知水平,重塑我们的未来? 这些问题的答案,或许都能在雷…...

NAS教程丨铁威马如何登录 SSH终端?
适用型号: 所有TNAS 型号 如您有特殊操作需要通过 SSH 终端登录 TNAS,请参照以下指引: (注意: 关于以下操作步骤中的"cd /"的指令,其作用是使当前 SSH/Telnet 连接的位置切换到根目录,以免造成对卷的占用.请不要遗漏它.) Windows…...

2024-06-24 百度地图的使用及gps定位坐标获取
1.百度地图的使用教程 2. 定位功能的实现 第一种:通过h5自带定位获取当前gps坐标 var options {enableHighAccuracy: true,timeout: 5000,maximumAge: 0};function success(pos) {var crd pos.coords;alert(crd.latitude---crd.longitude---crd.accuracy);conso…...

Python二级考试试题②
1. 以下关于程序设计语言的描述,错误的选项是: A Python语言是一种脚本编程语言 B 汇编语言是直接操作计算机硬件的编程语言 C 程序设计语言经历了机器语言、汇编语言、脚本语言三个阶段 D 编译和解释的区别是一次性翻译程序还是每次执行时都要翻…...

安装和使用nvm安装Nodejs
文章目录 安装和使用 nvm1. 安装 nvm2. 重新加载终端配置3. 安装所需的 Node.js 版本4. 使用安装的 Node.js 版本 nvm 常用命令 安装和使用 nvm 以下是安装 nvm 并使用它来安装 Node.js 的步骤: 1. 安装 nvm 首先,您需要安装 nvm。您可以使用 curl 或…...

非遗!四川省21市非遗大师工作室申报认定条件程序和认定补贴经费支持(管理办法)
第一章总则 第一条贯彻落实中共中央办公厅、国务院办公厅《关于进一步加强非物质文化遗产保护工作的意见》(厅字〔2021〕31号)、四川省文化和旅游厅等12部门《关于进一步加强非物质文化遗产保护工作的实施意见》(川文旅发〔2022〕25号&#…...