MySQL中什么情况下会出现索引失效?如何排查索引失效?

2024-04-26 07:20

本文主要是介绍MySQL中什么情况下会出现索引失效?如何排查索引失效?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

  • 1-引言:什么是MySQL的索引失效?(What、Why)
    • 1-1 索引失效定义
    • 1-2 为什么排查索引失效
  • 2- 索引失效的原因及排查(How)
    • 2-1 索引失效的情况
      • ① 索引列参与计算
      • ② 对索引列进行函数操作
      • ③ 查询中使用了 OR 两边有范围查询 > 或 <
      • ④ like 操作:以 % 开头的 like 查询
      • ⑤ 不等于比较 !=
      • ⑥ order by
      • ⑦ 使用 IN
    • 2-2 索引失效的排查
      • 使用 explain 排查
      • 需要关注的字段:type、key、extra
  • 3- 总结:索引失效知识点小结


image.png

1-引言:什么是MySQL的索引失效?(What、Why)

1-1 索引失效定义

  • 在MySQL中,索引是用来加快检索数据库记录的一种数据结构
  • 索引失效指的是在进行查询操作时,本应该使用索引来提升查询效率的场景下,数据库没有利用索引,而是采用了全表扫描的方式,这会大大增加查询时间和系统负担。

1-2 为什么排查索引失效

排查索引失效的原因是至关重要的,主要有以下方面:

  • 1. 提高查询效率:索引的主要目的是加快数据检索速度。当索引失效时,数据库系统可能退回到更慢的查询方法,如全表扫描,这会显著增加查询时间和降低整体性能。
  • 2. 降低服务器负载:使用索引可以显著减少数据库处理查询所需处理的数据量,从而减少CPU使用率和IO读写。如果索引失效,数据库必须加载更多数据,这会增加服务器的负载和资源消耗。

2- 索引失效的原因及排查(How)

2-1 索引失效的情况

  • 以以下的学生信息表举例
CREATE TABLE `student_info` (`student_id` int(11) NOT NULL AUTO_INCREMENT,`student_name` varchar(50) NOT NULL,`student_age` int(11) DEFAULT NULL,`enrollment_date` datetime DEFAULT NULL,PRIMARY KEY (`student_id`),UNIQUE KEY `student_name` (`student_name`),KEY `student_age` (`student_age`),KEY `enrollment_date` (`enrollment_date`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
  • 表的索引情况:
  • 总结来说,表 student_info 有四个字段上定义了索引:
    • 一个主键索引 student_id
    • 一个唯一索引 student_name
    • 以及两个普通索引 student_ageenrollment_date

image.png

① 索引列参与计算

  • 正常的通过 age 去做查询
    • 走的是 student_age 的索引
explain select * from student_info where student_age=21;

image.png

  • 如果索引列参与了计算进行查询
    • 索引失效
explain select * from student_info where student_age+1 =21;

image.png

  • 如果不是对列进行计算,而是对列等号右侧的值进行计算,结果还是走索引的。

image.png


② 对索引列进行函数操作

  • 正常的查询——>走索引
explain select * from student_info where enrollment_date = '2022-09-04 08:00:00';

image.png

  • 如果对查询的字段加上函数操作时,索引失效
explain select * from student_info where YEAR(enrollment_date) = 2022;

image.png


③ 查询中使用了 OR 两边有范围查询 > 或 <

  • 正常情况查询,查询使用 student_name 索引
explain select * from student_info where student_name='Helen' and student_age>15;

image.png

  • 如果使用了 OR 进行查询,两边包含范围查询 > 或 <
    • 此时索引失效
explain select * from student_info where student_name='Helen' or student_age>15;

image.png

  • 如果没有范围查询下使用 OR 还是正常走索引
explain select * from student_info where student_name='Helen' or student_age=18;

image.png


④ like 操作:以 % 开头的 like 查询

  • 以 % 开头的 LIKE 查询比如 LIKE ‘%abc’;;

⑤ 不等于比较 !=

  • 在MySQL中 != 比较有可能会导致不走索引,但如果对 id 进行 != 比较,是有可能走索引的。
  • != 比较是否走索引,与索引的选择、数据分布情况有关,不单是由于查询包含 != 而引起的。

⑥ order by

  • 如果使用 order by 时,表中的数据量很小,数据库会直接在内存中进行排序,而不使用索引

image.png


⑦ 使用 IN

  • 使用 IN 的时候,有可能走索引,也有可能不走索引。当在 IN 的取值范围比较大的时候有可能会导致索引失效,走全表扫描(NOT ININ的失效场景相同)。

2-2 索引失效的排查

使用 explain 排查

  • 和 MySQL 慢查询的排查类似,使用 Explain 语句来进行排查。

需要关注的字段:type、key、extra

  • 我们可以根据 key、type、extra 来判断一条语句是否走了索引。
  • 一般走索引的情况 :
    • key 值不为 null
    • type 值应该为 ref、eq_ref、range、const 这几个
    • extra 的话如果是 NULL,或者 using indedx,using index condition 都是可以的

索引失效情况

  • 如果一条语句出现了 type 值为 all、key 为 nullextra = Using where 此时是索引失效了

此时就需要排查索引失效的原因

    1. 索引是否符合最左前缀匹配
    1. 查询语句出现以上 7 种情况

3- 总结:索引失效知识点小结

MySQL中什么情况下会出现索引失效?如何排查索引失效?
回答
:::info
MySQL中索引失效的情况有

    1. 比如联合索引在查询的过程中不符合最左前缀原则,此时联合索引会失效
    1. 查询的语句 索引列 进行计算,此时会使得索引失效
    1. 查询的语句 对索引列进行了函数操作,比如利用了 **YEAR()** 函数
    1. 查询语句中 包含 **OR** ,且 OR 两侧有范围查询 也就是 **>****<** 此时索引会失效
    1. 查询语句中 使用了 **like****like** 中存在 以**%** 开头的匹配,此时索引会失效
    1. 查询语句中 使用了 **!=** 进行比较,但这种情况也和数据的分布情况有关系,
    1. 当数据表中的数据较少,使用 **order by** 的时候,可能会不走索引直接在内存中进行排序
    1. 当使用 **IN**** **时候取值范围比较大的时候有可能会导致索引失效

索引失效的排查

  • ① 使用 Explain 对 SQL 语句进行排查
  • 需要 关注的字段有 **type****key****extra**
  • 如果一条语句出现了 type 值为 all、key 为 nullextra = Using where 此时是索引失效了

此时就需要排查索引失效的原因,是否存在以上情况
:::

这篇关于MySQL中什么情况下会出现索引失效?如何排查索引失效?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/936940

相关文章

SQL中的CASE WHEN用法小结

《SQL中的CASEWHEN用法小结》文章详细介绍了SQL中的CASEWHEN函数及其用法,包括简单CASEWHEN和CASEWHEN条件表达式两种形式,并通过多个实际场景展示了如何使用CASEWH... 目录一、简单CASE WHEN函数:二、CASE WHEN条件表达式函数三、常用场景场景1:不同状态展

mysql删除无用用户的方法实现

《mysql删除无用用户的方法实现》本文主要介绍了mysql删除无用用户的方法实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 1、删除不用的账户(1) 查看当前已存在账户mysql> select user,host,pa

使用Dify访问mysql数据库详细代码示例

《使用Dify访问mysql数据库详细代码示例》:本文主要介绍使用Dify访问mysql数据库的相关资料,并详细讲解了如何在本地搭建数据库访问服务,使用ngrok暴露到公网,并创建知识库、数据库访... 1、在本地搭建数据库访问的服务,并使用ngrok暴露到公网。#sql_tools.pyfrom

MySQL中实现多表查询的操作方法(配sql+实操图+案例巩固 通俗易懂版)

《MySQL中实现多表查询的操作方法(配sql+实操图+案例巩固通俗易懂版)》本文主要讲解了MySQL中的多表查询,包括子查询、笛卡尔积、自连接、多表查询的实现方法以及多列子查询等,通过实际例子和操... 目录复合查询1. 回顾查询基本操作group by 分组having1. 显示部门号为10的部门名,员

MyBatis-Plus中静态工具Db的多种用法及实例分析

《MyBatis-Plus中静态工具Db的多种用法及实例分析》本文将详细讲解MyBatis-Plus中静态工具Db的各种用法,并结合具体案例进行演示和说明,具有很好的参考价值,希望对大家有所帮助,如有... 目录MyBATis-Plus中静态工具Db的多种用法及实例案例背景使用静态工具Db进行数据库操作插入

MySQL中COALESCE函数示例详解

《MySQL中COALESCE函数示例详解》COALESCE是一个功能强大且常用的SQL函数,主要用来处理NULL值和实现灵活的值选择策略,能够使查询逻辑更清晰、简洁,:本文主要介绍MySQL中C... 目录语法示例1. 替换 NULL 值2. 用于字段默认值3. 多列优先级4. 结合聚合函数注意事项总结C

通过ibd文件恢复MySql数据的操作方法

《通过ibd文件恢复MySql数据的操作方法》文章介绍通过.ibd文件恢复MySQL数据的过程,包括知道表结构和不知道表结构两种情况,对于知道表结构的情况,可以直接将.ibd文件复制到新的数据库目录并... 目录第一种情况:知道表结构第二种情况:不知道表结构总结今天干了一件大事,安装1Panel导致原来服务

mysql关联查询速度慢的问题及解决

《mysql关联查询速度慢的问题及解决》:本文主要介绍mysql关联查询速度慢的问题及解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录mysql关联查询速度慢1. 记录原因1.1 在一次线上的服务中1.2 最终发现2. 解决方案3. 具体操作总结mysql

Linux搭建Mysql主从同步的教程

《Linux搭建Mysql主从同步的教程》:本文主要介绍Linux搭建Mysql主从同步的教程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux搭建mysql主从同步1.启动mysql服务2.修改Mysql主库配置文件/etc/my.cnf3.重启主库my

MySql中的数据库连接池详解

《MySql中的数据库连接池详解》:本文主要介绍MySql中的数据库连接池方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录mysql数据库连接池1、概念2、为什么会出现数据库连接池3、原理4、数据库连接池的提供商5、DataSource数据源6、DBCP7、C