mysql如何计算查询代价_mysql优化成本分析

来源:这里教程网 时间:2026-02-28 20:46:50 作者:

怎么看 MySQL 实际执行的查询代价

MySQL 本身不直接暴露“查询代价”的数值(比如 12.7 cost 这种),

EXPLAIN
输出里的
cost
列仅在启用
optimizer_trace
或使用
EXPLAIN FORMAT=JSON
时才可能间接体现,且是优化器预估值,不是真实开销。真正可观察、可对比的代价,得看实际执行行为:

EXPLAIN
中的
rows
值——优化器预估扫描行数,越接近真实扫描量,预估越可信;明显偏高通常意味着统计信息过期或缺少有效索引
开启
profiling
(已弃用)或更推荐用
performance_schema
events_statements_history_long
,看
TIMER_WAIT
(纳秒级真实执行耗时)和
LOCK_TIME
SHOW PROFILE FOR QUERY N
(需先
SET profiling = 1
)看各阶段耗时分布,比如
Copying to tmp table
Sorting result
占比高,说明临时表或排序成了瓶颈

为什么
EXPLAIN
key_len
比预期小

key_len
显示的是 MySQL 实际用到的索引字节数,不是定义长度。它小,往往意味着索引没被充分利用,常见原因有:

索引列存在
NULL
,且查询条件是
IS NULL
或未加
NOT NULL
约束,导致无法使用最左前缀匹配
查询条件用了函数或表达式,如
WHERE YEAR(created_at) = 2024
,即使
created_at
有索引,
key_len
也会是 0
联合索引中,中间某列用了范围查询(
>
,
BETWEEN
,
LIKE 'abc%'
),后续列就失效,
key_len
只计算到范围列为止
字符集不同导致隐式转换,比如
utf8mb4
列与
utf8
字符串比较,索引可能部分失效

如何用
optimizer_trace
看优化器真实决策过程

这是唯一能看清 MySQL “怎么算出这个执行计划”的方式,但默认关闭,且有性能开销,只应在调试时临时启用:

执行
SET optimizer_trace="enabled=on", end_markers_in_json=on;
运行目标
SELECT
查询
SELECT * FROM information_schema.OPTIMIZER_TRACE;
,重点关注
steps
数组里的
considered_execution_plans
analyzing_range_alternatives
注意:每条 trace 只保留最后一次查询,且不会记录全表扫描以外的备选方案(除非强制
USE INDEX
别在生产环境长期开启——trace 本身会增加解析和记录开销,尤其复杂查询可能生成 MB 级 JSON

真实代价常被忽略的三个隐藏项

很多人只盯着

EXPLAIN
rows
和响应时间,但以下三项在高并发或大数据量下影响巨大:

Handler_read_next
/
Handler_read_rnd_next
:从
SHOW STATUS LIKE 'Handler%'
查,前者高说明索引扫描效率低,后者高基本等于频繁回表,是二级索引 +
SELECT *
的典型信号
临时表类型:若
Created_tmp_disk_tables
持续增长,说明
sort_buffer_size
tmp_table_size
不足,磁盘临时表比内存表慢 1–2 个数量级
锁等待:即使查询快,若
Innodb_row_lock_time_avg
超过 10ms,大概率存在热点行锁争用,这时优化 SQL 不如拆分更新粒度或调整事务边界

代价分析不是单看一条语句跑多快,而是把执行路径、资源消耗、并发干扰全串起来看——尤其是

Handler
状态和锁指标,它们比
EXPLAIN
更诚实。

相关推荐