Mysql 索引

本文详细介绍了MySQL索引的定义、类型和常见数据结构,如B树和B+树。讨论了索引的优缺点,强调了其在数据检索、排序和I/O优化中的作用。同时,讲解了主键索引、二级索引、唯一索引、普通索引、前缀索引和全文索引等不同类型的索引。此外,还分析了聚集索引与非聚集索引的区别,以及如何进行索引优化,包括避免索引失效的场景。最后,提出了索引创建的策略和注意事项,以及适合建索引的字段选择。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

MYSQL 官方对索引的定义为:

索引(Index)是帮助 MySQL 高效获取数据的数据结构,所以说索引的本质是:数据结构

常见的索引结构有: B 树, B+树和 Hash

索引的优缺点

优点
  • 索引大大减少了服务器需要扫描的数据量(提高数据检索效率)
  • 索引可以帮助服务器避免排序和临时表(降低数据排序的成本,降低CPU的消耗)
  • 索引可以将随机 I/O 变为顺序 I/O(降低数据库IO成本)
缺点
  • 索引需要使用物理文件存储,也会耗费一定空间
  • 修改数据时,如有索引,需要操作索引树,影响到 SQL 执行效率

B+树 索引

B 树也称 B-树,全称为 多路平衡查找树 ,B+ 树是 B 树的一种变体。B 树和 B+树中的 B 是 Balanced (平衡)的意思

B 树 & B+树
  1. B 树的所有节点既存放键(key) 也存放 数据(data),而 B+树只有叶子节点存放 key 和 data,其他内节点只存放 key
  2. B 树的叶子节点都是独立的,B+树的叶子节点有一条引用链指向与它相邻的叶子节点
  3. B 树的检索的过程相当于对范围内的每个节点的关键字做二分查找,可能还没有到达叶子节点,检索就结束了。而 B+树的检索效率就很稳定了,任何查找都是从根节点到叶子节点的过程,叶子节点的顺序检索很明显
    在这里插入图片描述

索引类型

主键索引

一张数据表有只能有一个主键,并且主键不能为 null,不能重复。

二级索引(辅助索引)

二级索引又称为辅助索引,是因为二级索引的叶子节点存储的数据是主键。也就是说,通过二级索引,可以定位主键的位置

  1. 唯一索引(Unique Key):唯一索引的属性列不能出现重复的数据,但是允许数据为 NULL,一张表允许创建多个唯一索引。 建立唯一索引的目的大部分时候都是为了该属性列的数据的唯一性,而不是为了查询效率
  2. 普通索引(Index) :普通索引的唯一作用就是为了快速查询数据,一张表允许创建多个普通索引,并允许数据重复和 NULL
  3. 前缀索引(Prefix) :前缀索引只适用于字符串类型的数据。前缀索引是对文本的前几个字符创建索引,相比普通索引建立的数据更小, 因为只取前几个字符
  4. 全文索引(Full Text) :全文索引主要是为了检索大文本数据中的关键字的信息,是目前搜索引擎数据库使用的一种技术。Mysql5.6 之前只有 MYISAM 引擎支持全文索引,5.6 之后 InnoDB 也支持了全文索引

唯一索引,普通索引,前缀索引等索引属于二级索引

聚集索引与非聚集索引

聚集索引

聚集索引即索引结构和数据一起存放的索引。主键索引属于聚集索引

聚集索引的查询速度非常的快,因为整个 B+树本身就是一颗多叉平衡树,叶子节点也都是有序的,定位到索引的节点,就相当于定位到了数据

聚集索引依赖有序数据,如果插入的主键值非有序(比如字符串或者UUID),插入速度和查找速度将比较慢。

非聚集索引

非聚集索引即索引结构和数据分开存放的索引。二级索引(辅助索引) 属于非聚集索引。

非聚集索引更新代价比聚集索引要小 ,非聚集索引的更新代价就没有聚集索引那么大了,非聚集索引的叶子节点是不存放数据的。

非聚集索引也依赖有序数据,由于叶子节点data域存储的是聚集索引的值,可能需要回表

非聚集索引一定会回表吗? (什么是覆盖索引?)

不一定。如果查询的那个字段刚好建立了索引,即不需要回表。
例如 SELECT name FROM table WHERE name = 'Mike';

覆盖索引即需要查询的字段正好是索引的字段,那么直接根据该索引,就可以查到数据了, 而无需回表查询。

索引创建策略

需要建索引的情况:

  • 频繁查询的字段
  • 查询中排序的字段,排序字段通过索引访问大幅提高排序速度
  • 查询中统计或分组字段

不需要创建索引的情况:

  • 表记录太少
  • 频繁更新的字段

索引优化

索引失效
  • 索引列不独立:被索引的这列不能是表达式的一部分,不能是函数的参数

例如下面的情况:
select id,name,age,salary from table_name where salary + 1000 = 6000;
解决:提前计算好条件值
select id,name,age,salary from table_name where salary = 5000;

索引字段作为函数的参数:
select id,name,age,salary from table_name where substring(name,1,3)= 'luc';
解决:like匹配
select id,name,age,salary from table_name where name like 'luc%';

  • 使用了左模糊

select id,name,age,salary from table_name where name like '%luc%';

  • 字符串条件没有使用 ‘’

select id,name,age,salary from table_name where phone=13088772233

  • 不符合最左前缀原则的查询

联合索引 Index(a,b,c)
select * from table_name where b='1'and c='2'
select * from table_name where c='2'
查询条件中没有 字段 a

  • 隐式转换

表关联字段类型不匹配

  • 使用 != 或<>操作符
建索引的注意事项

适合建索引的字段

  • 不为 NULL 的字段
  • 被频繁查询的字段
  • 被作为条件查询的字段
  • 频繁需要排序的字段
  • 被经常频繁用于连接的字段

被频繁更新的字段应该慎重建立索引

尽可能的考虑建立联合索引而不是单列索引

注意避免冗余索引

考虑在字符串类型的字段上使用前缀索引代替普通索引

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值