news 2026/6/1 9:52:46

MySQL索引

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
MySQL索引

MySQL索引详解

一、索引是什么?

索引是数据库中的一种数据结构,类似于书籍的目录,它可以帮助数据库快速定位和访问表中的数据,而不需要扫描整个表。

二、MySQL索引分类

1.按数据结构分类

索引类型存储引擎特点适用场景
B+Tree索引InnoDB、MyISAM默认索引,平衡树结构范围查询、排序、等值查询
Hash索引Memory/Heap哈希表结构,O(1)查找等值查询(不支持范围查询)
Full-Text索引InnoDB、MyISAM全文搜索文本字段的模糊搜索
R-Tree索引MyISAM空间数据索引GIS地理数据

2.按逻辑功能分类

主键索引 (Primary Key)
CREATETABLEusers(idINTPRIMARYKEY,-- 主键索引nameVARCHAR(50));
  • 每张表只能有一个
  • 不允许NULL值
  • 自动创建聚簇索引(InnoDB)
唯一索引 (Unique Index)
CREATEUNIQUEINDEXidx_emailONusers(email);
  • 确保列值唯一
  • 允许NULL值(但只能有一个NULL)
  • 可以有多个
普通索引 (Normal Index)
CREATEINDEXidx_nameONusers(name);
  • 最基本的索引类型
  • 仅用于加速查询
全文索引 (Full-Text Index)
CREATEFULLTEXTINDEXidx_contentONarticles(content);
  • 用于全文搜索
  • 支持MATCH AGAINST语法
组合索引 (Composite Index)
CREATEINDEXidx_name_ageONusers(name,age);
  • 多列组合的索引
  • 最左前缀原则
空间索引 (Spatial Index)
CREATESPATIALINDEXidx_locationONplaces(location);
  • 用于地理空间数据

3.按物理存储分类

聚簇索引 (Clustered Index)
  • InnoDB中主键就是聚簇索引
  • 数据行和索引存储在一起
  • 一个表只有一个
非聚簇索引 (Non-Clustered Index)
  • 索引和数据分开存储
  • MyISAM默认都是非聚簇索引
  • InnoDB的二级索引

三、常用索引类型

最常用的索引:

  1. B+Tree索引(95%以上场景)
  2. 组合索引(优化多条件查询)
  3. 唯一索引(保证数据唯一性)
  4. 主键索引(每张表必须有)

四、索引生效场景(什么时候有效)

1. 全值匹配

-- 索引 idx_name_age(name, age)SELECT*FROMusersWHEREname='张三'ANDage=25;-- ✅ 生效

2. 最左前缀匹配

-- 索引 idx_name_age_city(name, age, city)SELECT*FROMusersWHEREname='张三';-- ✅ 生效SELECT*FROMusersWHEREname='张三'ANDage=25;-- ✅ 生效SELECT*FROMusersWHEREage=25;-- ❌ 不生效(跳过了name)

3. 范围查询(部分生效)

-- 索引 idx_name_age(name, age)SELECT*FROMusersWHEREname='张三'ANDage>20;-- ✅ 生效SELECT*FROMusersWHEREname>'张三';-- ✅ 生效

4. 覆盖索引

-- 索引 idx_name_age(name, age)SELECTname,ageFROMusersWHEREname='张三';-- ✅ 生效(不需要回表)

5. 排序操作

-- 索引 idx_name_age(name, age)SELECT*FROMusersORDERBYname,age;-- ✅ 生效SELECT*FROMusersORDERBYnameASC,ageDESC;-- ❌ 不生效(混合排序)

6. 分组操作

-- 索引 idx_name_age(name, age)SELECTname,COUNT(*)FROMusersGROUPBYname;-- ✅ 生效

五、索引失效场景(什么时候无效)

1. 违反最左前缀原则

-- 索引 idx_name_age(name, age)SELECT*FROMusersWHEREage=25;-- ❌ 不生效SELECT*FROMusersWHEREage=25ANDname='张三';-- ✅ 生效(优化器会调整顺序)

2. 在索引列上运算或函数

-- 索引 idx_name(name)SELECT*FROMusersWHERELEFT(name,1)='张';-- ❌ 不生效SELECT*FROMusersWHEREYEAR(create_time)=2024;-- ❌ 不生效

3. 使用不等于(!=, <>)

SELECT*FROMusersWHEREname!='张三';-- ❌ 不生效(全表扫描更优)

4. 使用IS NULL/IS NOT NULL(特殊情况)

SELECT*FROMusersWHEREnameISNULL;-- ✅ 可能生效SELECT*FROMusersWHEREnameISNOTNULL;-- ❌ 通常不生效

5. LIKE以通配符开头

SELECT*FROMusersWHEREnameLIKE'%张三%';-- ❌ 不生效SELECT*FROMusersWHEREnameLIKE'张三%';-- ✅ 生效

6. 类型转换

-- 假设phone是varchar类型,索引 idx_phone(phone)SELECT*FROMusersWHEREphone=13800138000;-- ❌ 不生效(隐式类型转换)SELECT*FROMusersWHEREphone='13800138000';-- ✅ 生效

7. OR条件部分无索引

-- name有索引,age无索引SELECT*FROMusersWHEREname='张三'ORage=25;-- ❌ 不生效

8. 数据量小时

-- 表只有100行数据SELECT*FROMsmall_tableWHEREname='test';-- ❌ 可能不生效(全表扫描更快)

9. 统计信息不准确

-- 当索引统计信息过时,优化器可能选择全表扫描ANALYZETABLEusers;-- 更新统计信息

六、索引设计最佳实践

创建索引的黄金法则:

-- 1. 选择性高的列建索引CREATEINDEXidx_emailONusers(email);-- email唯一性高-- 2. 常用查询条件组合建索引CREATEINDEXidx_queryONorders(user_id,status,create_time);-- 3. 覆盖索引设计CREATEINDEXidx_coveringONusers(name,age,email);-- 查询:SELECT name, age, email FROM users WHERE name = '张三';

索引使用建议:

  1. 不要过度索引:每个索引都有维护成本
  2. 更新频繁的列:谨慎建索引
  3. 小表不建议建索引
  4. 避免冗余索引
  5. 定期分析索引使用情况
-- 查看索引使用情况SHOWINDEXFROMtable_name;-- 查看未使用的索引SELECT*FROMsys.schema_unused_indexes;

七、性能诊断工具

1. EXPLAIN分析

EXPLAINSELECT*FROMusersWHEREname='张三';

2. 查看索引统计

-- 查看索引区分度SELECTINDEX_NAME,CARDINALITY,TABLE_ROWS,ROUND(CARDINALITY/TABLE_ROWS*100,2)asselectivityFROMinformation_schema.STATISTICSWHERETABLE_NAME='users';

3. 慢查询日志

-- 开启慢查询SETGLOBALslow_query_log=ON;SETGLOBALlong_query_time=1;

八、常见问题排查

索引失效排查步骤:

  1. 使用EXPLAIN分析执行计划
  2. 检查WHERE条件是否符合最左前缀
  3. 检查是否有类型转换
  4. 检查是否使用函数或计算
  5. 检查统计信息是否准确
  6. 检查数据量是否太小

索引优化示例:

-- 问题查询SELECT*FROMordersWHEREDATE(create_time)='2024-01-01'-- ❌ 索引失效ANDstatus=1;-- 优化后SELECT*FROMordersWHEREcreate_time>='2024-01-01'ANDcreate_time<'2024-01-02'-- ✅ 索引生效ANDstatus=1;-- 创建合适索引CREATEINDEXidx_time_statusONorders(create_time,status);

记住原则:索引不是越多越好,合适的索引才是最好的。定期审查和优化索引是DBA的重要工作。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/1 1:06:28

2026 网络安全转行指南:零基础从入门到精通,一篇吃透

在当前就业形势下&#xff0c;不少朋友咨询过龙哥&#xff0c;询问转行能否转行网络安全。网络安全作为一个热门领域&#xff0c;自然也吸引了许多人的目光。本文将就转行网络安全这一话题&#xff0c;提供一些切实可行的建议。 网络安全行业概况 网络安全涵盖了从基础的脚本编…

作者头像 李华
网站建设 2026/5/28 18:29:07

Z-Image-Turbo_UI界面实战:批量生成商品图方案详解

Z-Image-Turbo_UI界面实战&#xff1a;批量生成商品图方案详解 你是否还在为电商运营中上千款SKU的商品图制作发愁&#xff1f;手动修图耗时、外包成本高、AI工具出图慢且风格不统一——这些痛点&#xff0c;正在拖慢你的上新节奏。今天要介绍的不是又一个“概念级”模型&…

作者头像 李华
网站建设 2026/5/30 15:16:26

YOLO11转RKNN全过程,图文并茂易理解

YOLO11转RKNN全过程&#xff0c;图文并茂易理解 本文是一份面向嵌入式AI开发者的实操指南&#xff0c;聚焦YOLO11模型从训练完成到部署至瑞芯微RK3588开发板的完整链路&#xff0c;尤其详述其中关键一环——ONNX模型向RKNN格式的转换过程。全文不堆砌理论&#xff0c;不罗列参…

作者头像 李华
网站建设 2026/5/29 16:09:35

Unsloth微调全攻略:支持Windows和Linux双平台

Unsloth微调全攻略&#xff1a;支持Windows和Linux双平台 1. 为什么你需要Unsloth——不是又一个微调框架&#xff0c;而是效率革命 你有没有试过在本地显卡上微调一个14B参数的模型&#xff1f;等了六个小时&#xff0c;显存爆了三次&#xff0c;最后发现训练出来的模型连基…

作者头像 李华
网站建设 2026/5/28 13:02:09

贴片LED灯正负极区分错误导致的硬件故障分析

以下是对您提供的博文内容进行 深度润色与结构重构后的技术类专业文章 。整体风格更贴近一位资深硬件工程师在技术社区或内刊中分享实战经验的口吻&#xff1a;语言精炼、逻辑严密、案例真实、有洞见、有温度&#xff0c;同时彻底去除AI生成痕迹和模板化表达&#xff0c;强化…

作者头像 李华
网站建设 2026/5/28 13:01:54

FSMN-VAD司法场景应用:审讯录音切分系统搭建

FSMN-VAD司法场景应用&#xff1a;审讯录音切分系统搭建 1. 为什么审讯录音需要“自动切分”&#xff1f; 你有没有想过&#xff0c;一份2小时的审讯录音&#xff0c;人工听写整理可能要花上一整天&#xff1f;更别说中间夹杂大量沉默、翻纸声、咳嗽、环境噪音——这些非语音…

作者头像 李华