数据库索引为什么会失效
本文来说下数据库的索引为什么会失效,本文以MySQL为例来说明下这个问题
文章目录
- MySQL数据是如何存储的
- 聚集索引
- 非聚集索引
- 联合索引/索引覆盖
- 索引下推
- 最左前缀原则
- 加速查询
- 加速排序
- 联合索引的好处
- 索引为什么会失效
- 本文小结
MySQL数据是如何存储的
聚集索引
我们先建如下的一张表
CREATE TABLE `student` (`id` int(11) NOT NULL AUTO_INCREMENT COMMENT '学号',`name` varchar(10) NOT NULL COMMENT '学生姓名',`age` int(11) NOT NULL COMMENT '学生年龄',PRIMARY KEY (`id`),KEY `idx_name` (`name`)
) ENGINE=InnoDB;
插入如下sql
insert into student (`name`, `age`) value('a', 10);
insert into student (`name`, `age`) value('c', 12);
insert into student (`name`, `age`) value('b', 9);
insert into student (`name`, `age`) value('d', 15);
insert into student (`name`, `age`) value('h', 17);
insert into student (`name`, `age`) value('l', 13);
insert into student (`name`, `age`) value('k', 12);
insert into student (`name`, `age`) value('x', 9);
数据如下
mysql是按照页来存储数据的,每个页的大小为16k。在MySQL中可以通过执行如下语句,看到一个页的大小
show global status like 'innodb_page_size'
结果为16384,即16kb
在InnoDB存储引擎中,是以主键为索引来组织数据的。记录在页中按照主键从小到大的顺序以单链表的形式连接在一起。
可能有小伙伴会问,如果建表的时候,没有指定主键呢?
如果在创建表时没有显示的定义主键,则InnoDB存储引擎会按如下方式选择或创建主键。
- 首先判断表中是否有非空的唯一索引,如果有,则该列即为主键。如果有多个非空唯一索引时,InnoDB存储引擎将选择建表时第一个定义的非空唯一索引作为主键
- 如果不符合上述条件,InnoDB存储引擎自动创建一个6字节大小的指针作为索引
页和页之间以双链表的形式连接在一起。并且下一个数据页中用户记录的主键值必须大于上一个数据页中用户记录的主键值
假设一个页只能存放3条数据,则数据存储结构如下。
可以看到我们想查询一个数据或者插入一条数据的时候,需要从最开始的页开始,依次遍历每个页的链表,效率并不高。
我们可以给这页做一个目录,保存主键和页号的映射关系,根据二分法就能快速找到数据所在的页。但这样做的前提是这个映射关系需要保存到连续的空间,如数组。如果这样做会有如下几个问题
- 随着数据的增多,目录所需要的连续空间越来越大,并不现实
- 当有一个页的数据全被删除了,则相应的目录项也要删除,它后面的目录项都要向前移动,成本太高
我们可以把目录数据放在和用户数据类似的结构中,如下所示。目录项有2个列,主键和页号。
数据很多时,一个目录项肯定很多,毕竟一个页的大小为16k,我们可以对数据建立多个目录项目,在目录项的基础上再建目录项,如下图所示
图片来自《MySQL 是怎样运行的:从根儿上理解 MySQL》。这其实就是一颗B+树,也是一个聚集索引,即数据和索引在一块。叶子节点保存所有的列值。
以 InnoDB 的一个整数字段索引为例,这个 N 差不多是 1200。这棵树高是 4 的时候,就可以存 1200 的 3 次方个值,这已经17 亿了。考虑到树根的数据块总是在内存中的,一个 10 亿行的表上一个整数字段的索引,查找一个值最多只需要访问 3次磁盘。其实,树的第二层也有很大概率在内存中,那么访问磁盘的平均次数就更少了。《MySQL实战45讲》
非聚集索引
聚集索引和非聚集索引非常类似,区别如下
聚集索引叶子节点的值为所有的列值非聚集索引叶子节点的值为索引列+主键
当我们查询name为h的用户信息时(学号,姓名,年龄),因为name上建了索引,先从name非聚集索引上,找到对应的主键id,然后根据主键id从聚集索引上找到对应的记录。
从非聚集索引上找到对应的主键值然后到聚集索引上查找对应记录的过程为回表
联合索引/索引覆盖
假设teacher表定义如下,在name和age列上建立联合索引
CREATE TABLE `teacher` (`id` int(11) NOT NULL AUTO_INCREMENT COMMENT '教师编号',`name` varchar(10) NOT NULL COMMENT '教师姓名',`age` int(11) NOT NULL COMMENT '教师年龄',`ismale` tinyint(3) NOT NULL COMMENT '是否男性',PRIMARY KEY (`id`),KEY `idx_name_age` (`name`, `age`)
) ENGINE=InnoDB;
插入如下sql
insert into teacher (`name`, `age`, `ismale`) value('aa', 10, 1);
insert into teacher (`name`, `age`, `ismale`) value('dd', 12, 0);
insert into teacher (`name`, `age`, `ismale`) value('cb', 9, 1);
insert into teacher (`name`, `age`, `ismale`) value('cb', 15, 1);
insert into teacher (`name`, `age`, `ismale`) value('bc', 17, 0);
insert into teacher (`name`, `age`, `ismale`) value('bb', 15, 1);
insert into teacher (`name`, `age`, `ismale`) value('dd', 15, 1);
insert into teacher (`name`, `age`, `ismale`) value('dd', 12, 0);
对name和age列建立联合索引
目录页由name列,age列,页号这三部分组成。目录会先按照name列进行排序,当name列相同的时候才对age列进行排序。
数据页由name列,age列,主键值这三部分组成。同样的,数据页会先按照name列进行排序,当name列相同的时候才对age列进行排序。
当执行如下语句的时候,会有回表的过程
select * from student where name = 'aa';
当执行如下语句的时候,没有回表的过程
select name, age from student where name = 'aa';
为什么不需要回表呢?
因为idx_name_age索引的叶子节点存的值为主键值,name值和age值,所以从idx_name_age索引上就能获取到所需要的列值,不需要回表,即索引覆盖
仔细看一下联合索引这个图,你就基本上能明白为什么不满足最左前缀原则的索引会失效?
索引下推
索引下推就是直接在索引上判断某些列值,不用回表再判断
当执行如下语句的时候
select * from student where name like '张%' and age = 10 and ismale = 1;
在5.6版本之前的执行过程如下,先从idx_name_age索引上找到对应的主键值,然后回表找到对应的行,判断其他字段的值是否满足条件
在5.6引入了索引下推优化,可以在遍历索引的过程中,对索引中包含的字段做判断,直接过滤掉不满足条件的数据,减少回表次数,如下图
最左前缀原则
加速查询
主要针对组合索引,满足如下2个条件即可满足左前缀原则
- 需要查询的列和组合索引的列顺序一致
- 查询不要跨列
构造数据如下,其中在name,address,country上建了联合索引
CREATE TABLE `people` (`name` varchar(50) NOT NULL,`address` varchar(50) NOT NULL,`country` varchar(50) NOT NULL,KEY `idx_name_addr_country` (`name`,`address`,`country`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
举几个例子,下面涉及到一些explain相关的知识,后面单开一篇长文来做介绍
例子一
explain select * from people where name = "jack"
and address = "beijing" and country = "china"
type为ref,key_len为456=(50*3+2)*3,联合索引的所有列都使用了
例子二
explain select * from people where name = "jack"
type为ref,key_len为152=50*3+2,联合索引只使用了name列
例子三
explain select * from people where address = "beijing"
type为index,表明查询的时候对整个索引进行了扫描,并没有加速查找。
假设有如下的联合索引 key idx_a_b_c(a,b,c)
sql | 是否使用索引 |
---|---|
where a = x and b = x and c = x | 是 |
where a = x and b = x | 是,部分索引 |
where a = x | 是,部分索引 |
where b = x | 否,不包含最左列name |
where b = x and c = x | 否,不包含最左列name |
如果你仔细看了前面联合索引是如何存储的,那你一定能看懂是否使用索引的介绍
目录页是按照 a b c 列的顺序依次递增排序的。先按照a列排序,如果a列相同,再按照b列排序,如果b列相同,才按照c列排序
所以查询列值a b c,则这个排序规则能用到,即会走索引。如果只查列值b,并不能用到这个排序规则,所以得遍历所有的记录
加速排序
最左前缀原则不仅用在查询中,还能用在排序中。MySQL中,有两种方式生成有序结果集:
- 通过有序索引顺序扫描直接返回有序数据
- Filesort排序,对返回的数据进行排序
因为索引的结构是B+树,索引中的数据是按照一定顺序进行排列的,所以在排序查询中如果能利用索引,就能避免额外的排序操作。EXPLAIN分析查询时,Extra显示为Using index。
所有不是通过索引直接返回排序结果的操作都是Filesort排序,也就是说进行了额外的排序操作。EXPLAIN分析查询时,Extra显示为Using filesort,当出现Using filesort时对性能损耗较大,所以要尽量避免Using filesort
还是先举2个例子,然后总结
explain select * from people order by name
Extra列只有Using index,即根据索引顺序进行扫描
explain select * from people order by address
Extra列有Using filesort
总结:假如说有如下联合索引,key idx_a_b_c(a,b,c)
order by 能使用索引排序
order by a
order by a,b
order by a,b,c
order by a desc, b desc, c desc
where a = const order by b,c
where a = const and b = const order by c
where a = const and b > const order by b,c
order by 不能使用索引进行排序
order by b
order by c
order by b, c
order by a asc, b desc, c desc //排序不一致
where g = const order by b,c //丢失a索引
where a = const order by c //丢失b索引
where a = const order by a,d //d不是索引的一部分
where a in (...) order by b,c //范围查询
这个原因就不用我解释了把,相信你一定看懂了
联合索引的好处
索引覆盖,减少了很多回表的操作,提高了查询的效率
索引下推,索引列越多,通过索引筛选出的数据越少。有1000W条数据的表,有如下sql:select * from table where col1=1 and col2=2 and col3=3,假设假设每个条件可以筛选出10%的数据,如果只有单值索引,那么通过该索引能筛选出1000W10%=100w条数据,然后再回表从100w条数据中找到符合col2=2 and col3= 3的数据;如果是联合索引,通过索引筛选出1000w*10%*10% *10%=1w,效率提升可想而知!
索引为什么会失效
当别人问我索引在什么条件下会失效时,我能背出一大堆规则
- 不要在索引列上进行运算或使用函数
- 前导模糊查询不会使用索引,例如 like %李
- 负向条件索引不会使用索引,建议用in。负向条件有:!=、<>、not in、not exists、not like 等
索引是按照一定规则排好序的,如果对索引列使用函数,或者 like % 李,具体的值都不知道,它怎么在B+树上加速查询?
本文小结
本文介绍了索引相关的知识,以及索引失效的原因,相信你会彻底掌握这个知识点。
数据库索引为什么会失效相关推荐
- oracle怎么找出失效索引,Oracle数据库索引失效
Oracle数据库中有一个表,用PL/SQL查看该表的索引没有被DROP掉, 但是表上的数据查询起来很慢(查询时间大概是原来的3倍),后 Oracle数据库中有一个表,用PL/SQL查看该表的索引没有 ...
- MySQL数据库索引的类型、命名规范、建立原则以及索引失效的情况
本篇中记录下数据库索引相关的知识点! 索引是什么? 举个例子:大家去图书馆借书时,会先在电脑检索书名或作者等关键字信息,查询出该本书对应的一个图书索引后,紧接着就可以拿着这个图书索引去精确定位存放该索 ...
- 范围查找(比如日期范围)下查询出现全表扫描MySQL数据库索引失效
范围查找(比如日期范围)下查询出现全表扫描MySQL数据库索引失效 当下MySQL数据库在多款数据库中脱颖而出,成为使用最广泛的数据库之一,这里我们来看看数据库索引上的一个问题.我们知道在数据量上去以 ...
- 数据库建表规约,索引创建及失效分析
文章目录 1. 建表规约 2. 索引规约 3. SQL 规约 1. 建表规约 1.1 表达是否概念的字段,使用 is_xxx 的方式命名,数据类型为 unsigned tinyint: 1.2 表名. ...
- Oracle数据库索引失效,引起GoldenGate异常
今天客户的GoldenGate出问题了,打电话过来咨询,根据客户描述的问题现象和日志的分析,初步判断是数据库索引异常引起的故障,以下是问题的分析和处理描述. 问题描述: 2010-02-01 17:1 ...
- 面试题:数据库索引失效的条件
1.不符合最左前缀法则(查询从索引的最左列开始不跳过中间的列) 2.范围查询右边的列不走索引 3.索引列上进行运算操作索引失效 4.字符串不加单引号,索引失效(底层进行运算操作) 5.使用or分隔开的 ...
- 数据库索引数据结构总结——ART树就是前缀树
数据库索引数据结构总结 from:https://zhewuzhou.github.io/2018/10/18/Database-Indexes/ 摘要 数据库索引是数据库中最重要的组成部分,而索引的 ...
- 什么是m叉树_不懂数据库索引的底层原理?那是因为你心里没点b树
点击上方"后端技术精选",选择"置顶公众号" 技术文章第一时间送达! 作者:苏苏喂 cnblogs.com/sujing/p/11110292.html 题外话 ...
- 数据库索引的实现原理及查询优化
MySQL官方对索引的定义为:索引(Index)是帮助MySQL高效获取数据的数据结构. 使用索引的目的在于提高查询效率,这篇文章梳理一下索引的实现原理和应用. 不同的存储引擎索引实现的数据结构不同 ...
最新文章
- 【机器视觉案例】(6) AI视觉,距离测量,自制AI小游戏,附python完整代码
- 【FFmpeg】警告:[hls] pkt.duration = 0, maybe the hls segment duration will not precise
- python面试题及答案-5个很好的Python面试题问题答案及分析
- GridView 序号 排序 正序 倒序
- jmeter mysql驱动jar包_用 Jmeter 做 Web 接口测试
- 新版python安装包(直接提取安装)
- golang rpc单参数调用实例
- Windows进程通信之共享内存通信(C++)
- Vmware虚拟机使用Nat方式连接笔记本无线网卡
- Q107:Mac系统下GDB对PBRT-V3进行debug
- 多速率多传感器数据融合估计(二)
- ArcEngine 渲染的使用【转载】
- EE214 Lecture 14 Differential pair部分公式推导
- html5播放器 php,PHP/HTML5页面上的随机声音播放器
- feign 实现签名、服务地址动态切换
- 风寒感冒一般低烧或者不发烧,清鼻涕。风热感冒一般有发烧比较厉害,黄鼻涕的反应
- 使用域控批量安装软件
- [译]如何打造以人为本的移动游戏
- js html url编码,js URLdecode()与urlencode方法支持中文解码
- WPF MVVM设计模式下 相同Xaml绑定不同ViewModel问题