索引可以理解为MySQL中用来高效检索数据的数据结构,它是有序的,因为它底层使用的数据结构是B+树;
从索引的设计原则上看,一般建议根据以下方式建立索引:
常见的有B+树索引结构和Hash索引结构,InnoDB存储引擎默认使用的就是B+树作为索引结构
B+树:数据都只存储在叶子节点中,而非叶子节点只用来进行索引,且叶子节点上维护了一条单向链表用来高效查询数据(InnoDB中维护的是双向链表);
至于InnoDB为什么使用B+树而不使用其它数据结构作为索引结构:
在MySQL数据库中,索引可以具体分为以下几类:
分类 | 含义 | 特点 | 关键字 |
---|---|---|---|
主键索引 | 针对于表中主键创造的索引 | 若有主键默认自动创建,且只有一个 | PRIMARY |
唯一索引 | 避免同一个表中某数据列中的值重复 | 可有多个 | UNIQUE |
常规索引 | 快速定位特定数据 | 可有多个 | |
全文索引 | 查找的是文本中的关键词,而本身比较索引中的值 | 可有多个 | FULLTEXT |
而在InnoDB存储引擎中,根据索引的存储格式可以分为以下两类:
分类 | 含义 | 特点 |
---|---|---|
聚集索引 | 索引结构的叶子节点存放行数据 | 必须有,且只能存在一个 |
二级索引 | 索引结构的叶子节点存放主键值 | 可有多个 |
对于聚集索引来说:
聚合索引和二级索引的具体结构如下:
可见,聚集索引叶子节点挂载的数据为主键值对应的行数据,而二级索引叶子节点挂载的数据为主键值;
根据查询字段不同可分为:
单列索引:一个索引中只包含一个字段;
联合索引:一个索引中包含多个字段;
一条SQL查询语句中涉及到了多个单列索引的话,只会执行其中的一个索引,而其它索引都会失效,对此,若业务需要涉及多个索引查询的话建议将这几个字段共同构建成一个联合索引:
create index idx_table_a_b_c on table(a, b, c);
联合索引的多个字段的索引值都存储在一起:
在使用联合查询时,需要遵守最左前缀法则,即sql语句中若使用到了联合索引中的字段,则该联合索引中的第一个字段必须出现,否则整个索引都会失效:
假设创建了一个联合索引如下:
create index idx_table_a_b_c on table(a, b, c);
则a
为联合索引中最左边的列,查询时必须存在,否则整个索引都会失效:
-- 有效索引
select * from table where a = 1 and b = 2 and c = 3
-- a不存在,无效索引
select * from table where b = 2 and c = 3;
-- 若中间一列被跳过,则从被跳过的那列开始后面的索引无效
select * from table where a = 1 and c = 3;
-- 有效索引,与顺序无关,只要第一个列存在即可实现索引
select * from table where c = 3 and b = 2 and a = 3;
注:若联合索引中出现范围查询(>
或 <
)的话,则范围查询右侧的索引都会失效,对此,若业务运行建议使用<=
或 >=
;
若需要查询的字段在该索引中都能找到,则称为覆盖索引,覆盖索引只需在二级索引中就能找到所需的数据并直接返回,无需执行回表查询,如图所示就是一个覆盖索引:
若需要查询的数据无法在二级索引中都找全,则需要进行回表查询:
像上述例子,则可以将其这几个字段共同构建成一个联合索引,则在进行数据索引时仅通过二级索引就能查找到所需的数据,避免回表查询:
create index idx_user_name_gender on user(name, gender);
当字段类型为字符串时,有时候需要索引很长的字符串(文本之类),这样会让索引变得很大,浪费大量磁盘IO,影响查询效率;此时可以只给字符串的一部分前缀建立索引,以此节约索引空间从而提高索引效率:
语法如下:
create index idx_table_a on table_name(column(n));
若一个字段存在多个索引(单列索引和联合索引),则在查询该字段时mysql内部会自动评估哪个执行效率更高进而使用哪个索引,可如果我们想要指定执行的索引的话,就可以使用SQL提示来让mysql执行我们指定的索引!
假设现在字段 a 有 单列索引idx_a
和 联合索引 idx_a_b_c
,我们想要它执行单列索引idx_a
,语法如下:
-- use index,建议mysql使用指定索引,但也只是建议,mysql内部还会进行再次评估
select * from table use index(idx_a) where a = 1;
-- ingonre index,忽略掉指定的索引
select * from table ignore index(idx_a_b_c) where a = 1;
-- force index,强制使用指定索引
select * from table force index(idx_a) where a = 1;
注:可以通过explain
关键字(加在sql语句之前)查询当前sql执行计划判断是否有使用到索引!
索引会在一些特殊情况会失效:
like '%三丰
’ 或 like '%三%'
),而尾部模糊匹配不会(like '张%'
);以上便是对MySQL 索引的介绍与使用了,如果上述内容对大家有帮助的话请给一个三连关注吧( •̀ ω •́ )✧✨