MySQL EXPLAIN中key_len使用的终极指南

 更新时间:2025年08月05日 10:29:34   作者:努力的小郑  
本文将深入解析MySQL执行计划中最关键的指标之一,EXPLAIN中key_len的使用,可以帮助大家快速定位索引优化点,从而提升查询性能

一、key_len:索引使用的精准标尺

在MySQL执行计划中,key_len表示查询实际使用索引的字节长度。这个指标是索引优化的核心,它能揭示:

  • 复合索引使用深度:显示使用了复合索引的前几列
  • 索引利用效率:值越大,索引利用率越高
  • 索引失效检测:NULL值表示索引未被使用
  • 数据类型成本:不同数据类型在索引中的开销
EXPLAIN SELECT * FROM orders WHERE user_id = 100 AND status = 'shipped';

-- 输出示例(修正对齐问题):
+----+-------------+--------+------+---------------+---------+---------+-------------+------+-------+
| id | select_type | table  | type | possible_keys | key     | key_len | ref         | rows | Extra |
+----+-------------+--------+------+---------------+---------+---------+-------------+------+-------+
|  1 | SIMPLE      | orders | ref  | idx_composite | idx_composite | 86      | const,const |    1 | NULL  |
+----+-------------+--------+------+---------------+---------+---------+-------------+------+-------+

二、key_len计算的核心规则(重点掌握!)

1. 基础计算规则

key_len = 数据类型基础长度 + NULL标记(1字节) + 变长类型额外开销

2. 常用数据类型计算表(utf8mb4环境)

数据类型基础长度NULL开销VARCHAR开销NOT NULL示例NULL示例
INT4字节+1字节-45
BIGINT8字节+1字节-89
TINYINT1字节+1字节-12
FLOAT4字节+1字节-45
DOUBLE8字节+1字节-89
DATE3字节+1字节-34
DATETIME8字节+1字节-89
TIMESTAMP4字节+1字节-45
CHAR(10)10×字符集字节+1字节-40 (utf8mb4)41 (utf8mb4)
VARCHAR(50)50×字符集字节+1字节+2字节202 (utf8mb4)203 (utf8mb4)
DECIMAL(10,2)整数4字节+小数2字节+1字节-67

核心要点

  • VARCHAR类型在索引中固定增加2字节长度前缀(实际行存储:≤255字符+1字节,>255字符+2字节)
  • 字符集直接影响长度:utf8mb4=4字节/字符,latin1=1字节/字符
  • NULL列增加1字节开销

三、key_len实战解析:从案例学优化

案例1:复合索引使用深度判断

-- 表结构
CREATE TABLE users (
  id INT PRIMARY KEY,
  name VARCHAR(50) NOT NULL,  -- key_len:50×4+2=202
  age TINYINT NOT NULL,        -- key_len:1
  email VARCHAR(100) NOT NULL, -- key_len:100×4+2=402
  INDEX idx_profile (name, age, email)
) CHARSET=utf8mb4;

-- 场景1:仅使用name列
EXPLAIN SELECT * FROM users WHERE name = 'John';
-- key_len = 202(复合索引第一列)

-- 场景2:使用前两列
EXPLAIN SELECT * FROM users WHERE name = 'John' AND age = 30;
-- key_len = 203(202+1)

-- 场景3:使用所有列
EXPLAIN SELECT * FROM users WHERE name = 'John' AND age = 30 AND email = 'john@example.com';
-- key_len = 605(202+1+402)

案例2:字符集对key_len的影响

-- latin1字符集对比
CREATE TABLE logs_latin1 (
  message VARCHAR(100) NOT NULL
) CHARSET=latin1;

CREATE TABLE logs_utf8mb4 (
  message VARCHAR(100) NOT NULL
) CHARSET=utf8mb4;

EXPLAIN SELECT * FROM logs_latin1 WHERE message = 'error';
-- key_len = 102 (100×1 + 2)

EXPLAIN SELECT * FROM logs_utf8mb4 WHERE message = 'error';
-- key_len = 402 (100×4 + 2)

案例3:NULL值的隐藏成本

-- 允许NULL的列
ALTER TABLE users MODIFY age TINYINT NULL;

-- 相同查询条件
EXPLAIN SELECT * FROM users WHERE name = 'John' AND age = 30;
-- key_len = 204(202+1+1,比非NULL多1字节)

四、key_len揭示的三大优化机会

1. 复合索引优化(核心!)

当key_len < 索引总长度时:

问题:索引未充分利用

解决方案

-- 1. 补充缺失查询条件
SELECT ... WHERE col1=1 AND col2=2 AND col3=3

-- 2. 重建索引(高频查询列前置)
ALTER TABLE orders DROP INDEX idx_old;
ALTER TABLE orders ADD INDEX idx_new (status, user_id, created_at);

-- 3. 使用覆盖索引
SELECT indexed_columns FROM table WHERE ...

2. VARCHAR列优化策略

-- 方案1:前缀索引(减少长度)
ALTER TABLE products ADD INDEX (description(20)); 
-- key_len从402降为82(VARCHAR(100)→20×4+2)

-- 方案2:ENUM替代(有限值场景)
ALTER TABLE orders 
  MODIFY status ENUM('pending','shipped','completed') NOT NULL;
-- key_len≈1-2字节

-- 方案3:压缩长文本+哈希索引
ALTER TABLE articles 
  ADD COLUMN content_hash BINARY(16) NOT NULL,
  ADD INDEX (content_hash);

3. 消除NULL存储开销

-- 优化前(允许NULL)
ALTER TABLE users MODIFY phone VARCHAR(20) NULL;
-- key_len=20×4+2+1=83

-- 优化后(禁止NULL)
ALTER TABLE users 
  MODIFY phone VARCHAR(20) NOT NULL DEFAULT '';
-- key_len=82(节省1字节/行)

五、高级诊断技巧

1. EXPLAIN FORMAT=JSON(推荐)

EXPLAIN FORMAT=JSON 
SELECT * FROM users WHERE name='Lisa';

/* 输出片段 */
{
  "query_block": {
    "table": {
      "key_length": 202,
      "used_key_parts": ["name"],
      // ...其他信息
    }
  }
}

2. 性能优化检查清单

  • 检查key_len是否接近索引长度
  • 确认复合索引是否满足最左前缀原则
  • 分析VARCHAR列长度是否合理
  • 检查是否有不必要的NULL列
  • 对比不同字符集下的索引大小

六、总结:key_len优化四原则

  • 追求最大key_len:值越接近索引总长度,索引利用越充分
  • 警惕NULL开销:每允许一个NULL列,key_len增加1字节
  • VARCHAR成本控制:长文本字段优先考虑前缀索引或哈希
  • 最左前缀原则:确保查询条件从复合索引最左侧开始

终极技巧:当发现key_len显著小于索引长度时,立即检查:

  • 是否缺少必要查询条件?
  • 索引列顺序是否合理?
  • 是否存在数据类型转换?
  • 字符集选择是否合适?

通过精准解读key_len,您将获得索引优化的"X光视力"。下次查看EXPLAIN结果时,重点关注key_len值——这个不起眼的数字可能是性能突破的关键!

到此这篇关于MySQL EXPLAIN中key_len使用的终极指南的文章就介绍到这了,更多相关MySQL EXPLAIN使用内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • MySQL中基本的用户和权限管理方法小结

    MySQL中基本的用户和权限管理方法小结

    这篇文章主要介绍了MySQL中基本的用户和权限管理方法小结,是MySQL入门学习中的基础知识,需要的朋友可以参考下
    2015-08-08
  • MySQL使用命令行备份数据的方法详解

    MySQL使用命令行备份数据的方法详解

    由于长期使用测试环境的数据库,时不时会有脏数据删除不干净,对此很需要一个实时将生产库的数据定期备份一份,防止生产库中会有脏数据进来,所以本文给大家介绍了MySQL使用命令行备份数据的方法,需要的朋友可以参考下
    2024-02-02
  • Mysql忘记root密码怎么办

    Mysql忘记root密码怎么办

    MySQL有时候忘记了root密码是一件伤感的事,下面通过本文给大家介绍Mysql忘记root密码怎么办的相关知识,需要的朋友参考下
    2016-01-01
  • MySQL中联表更新与删除的语法介绍

    MySQL中联表更新与删除的语法介绍

    这篇文章主要给大家介绍了MySQL中联表更新与删除的语法,文中通过示例代码介绍的很详细,有需要的朋友可以参考借鉴,下面来跟着小编一起学习学习吧。
    2017-01-01
  • mysql存储过程原理与用法详解

    mysql存储过程原理与用法详解

    这篇文章主要介绍了mysql存储过程,结合图文与实例形式较为详细的分析了mysql存储过程基本概念、原理、用法及相关操作注意事项,需要的朋友可以参考下
    2020-04-04
  • 获取缺失主键表信息的MYSQL语句

    获取缺失主键表信息的MYSQL语句

    如何获取缺失主键表信息,想必有些朋友还是不会的吧,下面为大家介绍下,可以通过几条语句实现,感兴趣的朋友可以了解下
    2013-06-06
  • MySQL转换Oracle的需要注意的七个事项

    MySQL转换Oracle的需要注意的七个事项

    有很多应用项目, 刚起步的时候用MySQL数据库基本上能实现各种功能需求,随着应用用户的增多,数据量的增加,MySQL渐渐地出现不堪重负的情况:连接很慢甚至宕机,于是就有MySQL转换Oracle的需求,应用程序也要相应做一些修改。
    2010-12-12
  • 解决MySQL登录报错1045-Access denied for user 'root'@' '(using password:YES)

    解决MySQL登录报错1045-Access denied for user 'root'@

    这篇文章主要给大家介绍了关于解决MySQL登录报错1045-Access denied for user ‘root‘@‘‘(using password:YES)的相关资料,文中一步步将解决的办法介绍的非常详细,需要的朋友可以参考下
    2023-07-07
  • MySQL进行监控配置的详细指南

    MySQL进行监控配置的详细指南

    这篇文章主要和大家分享一个完整的 MySQL 监控配置实战方案,涵盖监控工具安装,核心指标采集,可视化展示,告警配置等内容,需要的可以了解下
    2025-06-06
  • MySql恢复数据方法梳理讲解

    MySql恢复数据方法梳理讲解

    MySQL的binlog日志是MySQL日志中非常重要的一种日志,下面这篇文章主要给大家介绍了关于mysql如何利用binlog进行数据恢复的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考下
    2022-08-08

最新评论