`
jveqi
  • 浏览: 326518 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

MySQL索引之B+Tree

 
阅读更多

正确的创建合适的索引,是提升数据库查询性能的基础。在正式讲解之前,对后面举例中使用的表结构先简单看一下:

create table user
(
    id     bigint  not null comment 'id' primary key,
    name   varchar(200) null comment 'name',
    age    bigint       null comment 'age',
    gender int          null comment 'gender',
    key (name)
);
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8

1 索引是什么及工作机制?

索引是为了加速对表中数据行的检索而创建的一种分散存储的数据结构。其工作机制如下图:
在这里插入图片描述

上图中,如果现在有一条sql语句 select * from user where id = 40,如果没有索引的条件下,我们要找到这条记录,我们就需要在数据中进行全表扫描,匹配id = 13的数据。

如果有了索引,我们就可以通过索引进行快速查找,如上图中,可以先在索引中通过id = 40进行二分查找,再根据定位到的地址取出对应的行数据。

2. MySQL数据库为什么要使用B+TREE作为索引的数据结构?

2.1 二叉树为什么不可行

对数据的加速检索,首先想到的就是二叉树,二叉树的查找时间复杂度可以达到O(log2(n))。下面看一下二叉树的存储结构:
在这里插入图片描述

二叉树搜索相当于一个二分查找。二叉查找能大大提升查询的效率,但是它有一个问题:二叉树以第一个插入的数据作为根节点,如上图中,如果只看右侧,就会发现,就是一个线性链表结构。如果我们现在的数据只包含1, 2, 3, 4,就会出现以下情况:在这里插入图片描述

如果我们要查询的数据为4,则需要遍历所有的节点才能找到4,即,相当于全表扫描,就是由于存在这种问题,所以二叉查找树不适合用于作为索引的数据结构。

2.2 平衡二叉树为什么不可行

为了解决二叉树存在线性链表的问题,会想到用平衡二叉查找树来解决。下面看看平衡二叉树是怎样的:
在这里插入图片描述
平衡二叉查找树定义为:节点的子节点高度差不能超过1,如上图中的节点20,左节点高度为1,右节点高度0,差为1,所以上图没有违反定义,它就是一个平衡二叉树。保证二叉树平衡的方式为左旋,右旋等操作,至于如何左旋右旋,可以自行去搜索相关的知识。

如果上图中平衡二叉树保存的是id索引,现在要查找id = 8的数据,过程如下:

  1. 把根节点加载进内存,用8和10进行比较,发现8比10小,继续加载10的左子树。
  2. 把5加载进内存,用8和5比较,同理,加载5节点的右子树。
  3. 此时发现命中,则读取id为8的索引对应的数据。

索引保存数据的方式一般有两种:

  • 数据区保存id 对应行数据的所有数据具体内容。
  • 数据区保存的是真正保存数据的磁盘地址。

到这里,平衡二叉树解决了存在线性链表的问题,数据查询的效率好像也还可以,基本能达到O(log2(n)), 那为什么mysql不选择平衡二叉树作为索引存储结构,他又存在什么样的问题呢?

  1. 搜索效率不足。一般来说,在树结构中,数据所处的深度,决定了搜索时的IO次数(MySql中将每个节点大小设置为一页大小,一次IO读取一页 / 一个节点)。如上图中搜索id = 8的数据,需要进行3次IO。当数据量到达几百万的时候,树的高度就会很恐怖。
  2. 查询不不稳定。如果查询的数据落在根节点,只需要一次IO,如果是叶子节点或者是支节点,会需要多次IO才可以。
  3. 存储的数据内容太少。没有很好利用操作系统和磁盘数据交换特性,也没有利用好磁盘IO的预读能力。因为操作系统和磁盘之间一次数据交换是以页为单位的,一页大小为 4K,即每次IO操作系统会将4K数据加载进内存。但是,在二叉树每个节点的结构只保存一个关键字,一个数据区,两个子节点的引用,并不能够填满4K的内容。幸幸苦苦做了一次的IO操作,却只加载了一个关键字。在树的高度很高,恰好又搜索的关键字位于叶子节点或者支节点的时候,取一个关键字要做很多次的IO。

那有没有一种结构能够解决二叉树的这种问题呢?有,那就是多路平衡查找树。

2.3 多路平衡查找树(Balance Tree)

B Tree 是一个绝对平衡树,所有的叶子节点在同一高度,如下图所示:
在这里插入图片描述
上图为一个2-3树(每个节点存储2个关键字,有3路),多路平衡查找树也就是多叉的意思,从上图中可以看出,每个节点保存的关键字的个数和路数关系为:关键字个数 = 路数 – 1。

假设要从上图中查找id = X的数据,B TREE 搜索过程如下:

  1. 取出根磁盘块,加载40和60两个关键字。
  2. 如果X等于40,则命中;如果X小于40走P1;如果40 < X < 60走P2;如果X = 60,则命中;如果X > 60走P3。
  3. 根据以上规则命中后,接下来加载对应的数据, 数据区中存储的是具体的数据或者是指向数据的指针。

为什么说这种结构能够解决平衡二叉树存在的问题呢?

B Tree 能够很好的利用操作系统和磁盘的交互特性, MySQL为了很好的利用磁盘的预读能力,将页大小设置为16K,即将一个节点(磁盘块)的大小设置为16K,一次IO将一个节点(16K)内容加载进内存。这里,假设关键字类型为 int,即4字节,若每个关键字对应的数据区也为4字节,不考虑子节点引用的情况下,则上图中的每个节点大约能够存储(16 * 1000)/ 8 = 2000个关键字,共2001个路数。对于二叉树,三层高度,最多可以保存7个关键字,而对于这种有2001路的B树,三层高度能够搜索的关键字个数远远的大于二叉树。

这里顺便说一下:在B Tree保证树的平衡的过程中,每次关键字的变化,都会导致结构发生很大的变化,这个过程是特别浪费时间的,所以创建索引一定要创建合适的索引,而不是把所有的字段都创建索引,创建冗余索引只会在对数据进行新增,删除,修改时增加性能消耗。

B树确实已经很好的解决了问题,我先这里先继续看一下B+Tree结构,再来讨论BTree和B+Tree的区别。

先看看B+Tree是怎样的,B+Tree是B Tree的一个变种,在B+Tree中,B树的路数和关键字的个数的关系不再成立了,数据检索规则采用的是左闭合区间,路数和关键个数关系为1比1,具体如下图所示:
在这里插入图片描述
如果上图中是用ID做的索引,如果是搜索X = 1的数据,搜索规则如下:

  1. 取出根磁盘块,加载1,28,66三个关键字。
  2. X <= 1 走P1,取出磁盘块,加载1,10,20三个关键字。
  3. X <= 1 走P1,取出磁盘块,加载1,8,9三个关键字。
  4. 已经到达叶子节点,命中1,接下来加载对应的数据,图中数据区中存储的是具体的数据。

2.4 B TREE和B+TREE区别是什么?

  1. B+Tree 关键字的搜索采用的是左闭合区间,之所以采用左闭合区间是因为他要最好的去支持自增id,这也是mysql的设计初衷。即,如果id = 1命中,会继续往下查找,直到找到叶子节点中的1。

  2. B+Tree 根节点和支节点没有数据区,关键字对应的数据只保存在叶子节点中。即只有叶子节点中的关键字数据区才会保存真正的数据内容或者是内容的地址。而在B树种,如果根节点命中,则会直接返回数据。

  3. 在B+Tree中,叶子节点不会去保存子节点的引用。

  4. B+Tree叶子节点是顺序排列的,并且相邻的节点具有顺序引用的关系,如上图中叶子节点之间有指针相连接。

2.5 MySQL为什么最终要去选择B+Tree?

  1. B+Tree是B TREE的变种,B TREE能解决的问题,B+TREE也能够解决(降低树的高度,增大节点存储数据量)

  2. B+Tree扫库和扫表能力更强。如果我们要根据索引去进行数据表的扫描,对B TREE进行扫描,需要把整棵树遍历一遍,而B+TREE只需要遍历他的所有叶子节点即可(叶子节点之间有引用)。

  3. B+TREE磁盘读写能力更强。他的根节点和支节点不保存数据区,所以根节点和支节点同样大小的情况下,保存的关键字要比B TREE要多。而叶子节点不保存子节点引用,能用于保存更多的关键字和数据。所以,B+TREE读写一次磁盘加载的关键字比B TREE更多。

  4. B+Tree排序能力更强。上面的图中可以看出,B+Tree天然具有排序功能。

  5. B+Tree查询性能稳定。B+Tree数据只保存在叶子节点,每次查询数据,查询IO次数一定是稳定的。当然这个每个人的理解都不同,因为在B TREE如果根节点命中直接返回,确实效率更高。

3 MySQL B+Tree具体落地形式

这里主要讲解的是MySQL根据B+Tree索引结构不同的两种存储引擎(MYISAM 和 INNODB)的实现。

首先找到MySQL保存数据的文件夹,看看MySQL是如何保存数据的:

mysql> show variables like '%datadir%';
+---------------+------------------------+
| Variable_name | Value                  |
+---------------+------------------------+
| datadir       | /usr/local/mysql/data/ |
+---------------+------------------------+
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6

进入到这个目录下,这个目录下保存的是所有数据库,再进入到具体的一个数据库目录下。就能够看到MySQL存储数据和索引的文件了。

这里我创建了两张表,user_innod和user_myisam,分别指定索引为innodb和myisam。对于每张表,MySQL会创建相应的文件保存数据和索引,具体如下:

-rw-rw----. 1 mysql mysql      8652 May  3 21:11 user_innodb.frm
-rw-rw----. 1 mysql mysql 109051904 May  7 21:26 user_innodb.ibd
-rw-rw----. 1 mysql mysql      8682 May 16 18:27 user_myisam.frm
-rw-rw----. 1 mysql mysql         0 May 16 18:27 user_myisam.MYD
-rw-rw----. 1 mysql mysql      1024 May 16 18:27 user_myisam.MYI
  • 1
  • 2
  • 3
  • 4
  • 5

从图中可以看出:

  • MYISAM存储引擎存储数据库数据,一共有三个文件:
    Frm:表的定义文件。
    MYD:数据文件,所有的数据保存在这个文件中。
    MYI:索引文件。

  • Innodb存储引擎存储数据库数据,一共有两个文件(没有专门保存数据的文件):
    Frm文件: 表的定义文件。
    Ibd文件:数据和索引存储文件。数据以主键进行聚集存储,把真正的数据保存在叶子节点中。

3.1 MyISAM存储引擎

说明:为了画图简便,下面部分图使用在线数据结构工具进行组织数据,组织的B+Tree为右闭合区间,但不影响理解存储引擎数据存储结构。

在MYISAM存储引擎中,数据和索引的关系如下:

在这里插入图片描述

如何查找数据的呢?
如果要查询id = 40的数据:先根据MyISAM索引文件(如上图左)去找id = 40的节点,通过这个节点的数据区拿到真正保存数据的磁盘地址,再通过这个地址从MYD数据文件(如上图右)中加载对应的记录。

如果有多个索引,表现形式如下:
在这里插入图片描述
所以在MYISAM存储引擎中,主键索引和辅助索引是同级别的,没有主次之分。

3.2 Innodb存储引擎

Innodb主键索引为聚集索引,首先简单理解一下聚集索引的概念:数据库表行中数据的物理顺序和键值的逻辑顺序相同。

Innodb以主键索引来聚集组织数据的存储,下面看看Innodb是如何组织数据的。

在这里插入图片描述
如上图中,叶子节点的数据区保存的就是真实的数据,在通过索引进行检索的时候,命中叶子节点,就可以直接从叶子节点中取出行数据。mysql5.5版本之前默认采用的是MyISAM引擎,5.5之后默认采用的是innodb引擎。

在innodb中,辅助索引的格式如下图所示?
在这里插入图片描述

如上图,主键索引的叶子节点保存的是真正的数据。而辅助索引叶子节点的数据区保存的是主键索引关键字的值。

假如要查询name = C 的数据,其搜索过程如下:

  1. 先在辅助索引中通过C查询最后找到主键id = 9.
  2. 在主键索引中搜索id为9的数据,最终在主键索引的叶子节点中获取到真正的数据。

所以通过辅助索引进行检索,需要检索两次索引。

之所以这样设计,一个原因就是:如果和MyISAM一样在主键索引和辅助索引的叶子节点中都存放数据行指针,一旦数据发生迁移,则需要去重新组织维护所有的索引。

把Innodb 和 MYISAM区别放在一张图中看,就如下所示:

在这里插入图片描述

4 创建索引的几大原则

4.1 列的离散型

离散型的计算公式:count(distinct column_name):count(*),就是用去重后的列值个数比个数。值在 (0,1] 范围内。离散型越高,选择型越好。

如下表中各个字段,明显能看出Id的选择性比gender更高。

mysql> select * from user;
+----+--------------+------+--------+
| id | name         | age  | gender |
+----+--------------+------+--------+
| 20 | 君莫笑       |   15 |      1 |
| 40 | 苏沐橙       |   12 |      0 |
| 50 | 张楚岚       |   25 |      1 |
| 60 | 诸葛青       |   27 |      1 |
| 61 | 若有人兮     |   38 |      0 |
| 64 | 冯宝宝       |   18 |      0 |
+----+--------------+------+--------+
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

为什么说离散型越高,选择型越好?
因为离散度越高,通过索引最终确定的范围越小,最终扫面的行数也就越少。

4.2 最左匹配原则

对于索引中的关键字进行对比的时候,一定是从左往右以此对比,且不可跳过。之前讲解的id都为int型数据,如果id为字符串的时候,如下图:
在这里插入图片描述
当进行匹配的时候,会把字符串转换成ascll码,如abc变成97 98 99,然后从左往右一个字符一个字符进行对比。所以在sql查询中使用like %a 时候索引会失效,因为%表示全匹配,如果已经全匹配就不需要索引,还不如直接全表扫描。

4.3 最少空间原则

前面已经说过,当关键字占用的空间越小,则每个节点保存的关键字个数就越多,每次加载进内存的关键字个数就越多,检索效率就越高。创建索引的关键字要尽可能占用空间小。

5 联合索引

单列索引:节点中的关键字[name]
联合索引:节点中的关键字[name, age]

可以把单列索引看成特殊的联合索引,联合索引的比较也是根据最左匹配原则。

5.1 联合索引列的选择原则

  • 经常用的列优先(最左匹配原则)
  • 离散度高的列优先(离散度高原则)
  • 宽度小的列优先(最少空间原则)

5.2 实例分析

下面简单举例平时经常会遇到的问题:
如,平时经常使用的查询sql如下:
select * from users where name = ?
select * from users where name = ? and age = ?

为了加快检索速度,为上面的查询sql创建索引如下:
create index idx_name on users(name)
create index idx_name_age on users(name, age)

在上面解决方案中,根据最左匹配原则,idx_name为冗余索引, where name = ?同样可以利用索引idx_name_age进行检索。冗余索引会增加维护B+TREE平衡时的性能消耗,并且占用磁盘空间。

6. 覆盖索引

如果查询的列,通过索引项的信息可直接返回,则该索引称之为查询SQL的覆盖索引。覆盖索引可以提高查询的效率。
在这里插入图片描述
如上图,如果通过name进行数据检索:
select * from users where name = ?
需要需要在name索引中找到name对应的Id,然后通过获取的Id在主键索引中查到对应的行。整个过程需要扫描两次索引,一次name,一次id。

如果我们查询只想查询id的值,就可以改写SQL为:
select id from users where name = ?
因为只需要id的值,通过name查询的时候,扫描完name索引,我们就能够获得id的值了,所以就不需要再去扫面id索引,就会直接返回。

当然,如果你同时需要获取age的值:
select id,age from users where name = ?
这样就无法使用到覆盖索引了。

知道了覆盖索引,就知道了为什么sql中要求尽量不要使用select *,要写明具体要查询的字段。其中一个原因就是在使用到覆盖索引的情况下,不需要进入到数据区,数据就能直接返回,提升了查询效率。在用不到覆盖索引的情况下,也尽可能的不要使用select *,如果行数据量特别多的情况下,可以减少数据的网络传输量。当然,这都视具体情况而定,通过select返回所有的字段,通用性会更强,一切有利必有弊。

7 总结

  • 索引列的数据长度满足业务的情况下能少则少。

  • 表中的索引并不是越多越好,冗余或者无用索引会占用磁盘空间并且会影响增删改的效率。

  • Where 条件中,like 9%, like %9%, like%9,三种方式都用不到索引。后两种方式对于索引是无效的。第一种9%是不确定的,决定于列的离散型,结论上讲可以用到,如果发现离散情况特别差的情况下,查询优化器觉得走索引查询性能更差,还不如全表扫描。

  • Where条件中IN可以使用索引, NOT IN 无法使用索引。

  • 多用指定查询,只返回自己想要的列,少用select *。

  • 查询条件中使用函数,索引将会失效,这和列的离散性有关,一旦使用到函数,函数具有不确定性。

  • 联合索引中,如果不是按照索引最左列开始查找,无法使用索引。

  • 对联合索引精确匹配最左前列并范围匹配另一列,可以使用到索引。

  • 联合索引中,如果查询有某个列的范围查询,其右边所有的列都无法使用索引。

分享到:
评论

相关推荐

    MySQL - B+Tree伪图

    B+Tree是数据库管理系统中常用的一种数据结构,尤其在MySQL等关系型数据库中,B+Tree常被用于索引存储。B+Tree的特性使其在大规模数据存储中表现出优秀的查找、插入和删除效率。 **B+Tree的基本概念:** 1. **节点...

    为什么MySQL使用B+Tree1

    MySQL选择使用B+Tree作为其主要的数据存储结构,原因在于B+Tree的高效查询和适应磁盘I/O的特性。在数据库系统中,查询和...因此,B+Tree成为MySQL等数据库系统中理想的索引结构,尤其适合处理大规模数据的存储和检索。

    mysql mysql索引B+树 mysql面试题知识点

    B+Tree 索引是多数 MySQL 存储引擎默认采用的索引类型。 **优点**: - **快速查找**:通过构建树形结构,可以显著加快数据查找的速度。 - **排序和分组**:索引不仅可以用于查找特定的数据,还可以用于排序和分组...

    B+ Tree 增删改查 可视化

    B+ Tree被广泛应用于数据库索引,如MySQL的InnoDB存储引擎就使用了B+ Tree。文件系统如ext4也采用B+ Tree作为文件系统的i-node索引结构,以优化文件的查找和读取速度。 综上所述,B+ Tree是一种高效的索引数据结构...

    B+ tree的java实现和C++实现

    在实际应用中,B+树常用于数据库索引,例如MySQL的InnoDB存储引擎就使用了B+树作为索引结构。此外,在文件系统中,B+树也被用于快速定位和读取文件数据。 总结来说,B+树是一种重要的数据结构,其Java和C++实现都...

    MySQL索引背后的数据结构及算法原理

    正如标题所提到的,“MySQL索引背后的数据结构及算法原理”这一主题是技术面试中的重要内容之一。本文旨在深入探讨MySQL索引的相关概念及其背后的原理。 #### 索引的本质 索引本质上是一种数据结构,其目的是帮助...

    1,int(20)中20的涵义 2,为什么索引结构默认使用B+Tree,而不是Hash,二叉树,红黑树? 3、MySQL里记录

    2,为什么索引结构默认使用B+Tree,而不是Hash,二叉树,红黑树? 3、MySQL里记录货币用什么字段类型好 4、数据库自增主键可能遇到什么问题。 5、从锁的类别角度讲,MySQL都有哪些锁呢? 6、索引失效情况? 7、优化...

    B+Tree1.0.zip

    B+树在数据库索引、文件系统、地图索引等领域有着广泛应用,例如MySQL的InnoDB存储引擎就使用了B+树作为索引结构。它的优势在于: 1. 高效的范围查询:由于所有数据都在叶子节点,可以快速地进行区间查询。 2. 平衡...

    mysql索引原理之聚簇索引1

    相比之下,InnoDB引擎采用聚簇索引,主键的值直接作为B+Tree叶子节点的数据,数据行与索引在同一结构中。非聚簇索引的叶子节点则存储主键值,而非数据行地址。InnoDB的这种设计提高了范围查询的效率,因为它可以避免...

    MySQL的索引-你真的了解了吗

    MySQL 索引详解 MySQL 索引是数据库管理系统中一个排序的数据结构,用于协助快速查询、更新数据库表中数据。...因此,MySQL 选择了 B+Tree 作为索引,B+Tree 是一种多级索引结构,它可以减少 IO 次数,提高查询效率。

    B+Tree索引的背后

    本文主要探讨了MySQL数据库中InnoDB存储引擎的B+Tree索引机制。MySQL作为一个支持多种存储引擎的数据库系统,不同的引擎对于索引的支持各有特点。InnoDB因其广泛的应用和良好的性能特性,成为了讨论的重点。 **聚集...

    Mysql索引数据结构.pptx

    4. **B+ Tree**:B+ Tree 是 B Tree 的变体,更适用于数据库索引。与 B Tree 不同,B+ Tree 的非叶子节点不存储实际数据,只存储索引,这样可以容纳更多的索引项。叶子节点包含了所有索引字段,并且用指针互相连接,...

    MySQL 索引:索引为什么使用 B+树? · .pdf

    首先,我们从二叉查找树(Binary Search Tree,BST)开始,逐步说明各种树解决的问题以及面临的新问题,从而解释 MySQL 为什么选择 B+ 树作为索引结构。 二叉查找树(BST) 二叉查找树是一种常见的选择,因为它...

    mysql存储与索引技术

    B-Tree 适用于平衡查找,而 B+Tree 更适合数据库索引,因为它所有的数据都在叶子节点,且叶子节点之间有指针链接,这使得范围查询更为高效。 MySQL 的索引分为两种主要类型:MyISAM 使用非聚集索引,索引与数据分开...

    MySQL索引背后的数据结构及算法原理-07071521.pdf

    本文档主要讨论B-Tree索引,具体包括B-Tree和B+Tree的数据结构特点、MySQL索引的实现、索引使用策略及优化等。 MyISAM和InnoDB是MySQL中的两种主要存储引擎,它们对索引的实现各不相同。MyISAM存储引擎使用的是非...

    MySQL Innodb 索引原理详解

    在深入探讨MySQL Innodb索引之前,我们先了解几种基本的树形数据结构,包括二叉搜索树、B树、B+树以及B*树。 ##### 1.1 搜索二叉树(Binary Search Tree) 搜索二叉树是一种特殊的二叉树,每个节点至多有两个子...

    mysql索引原理深入解析

    总之,MySQL索引的原理在于利用各种数据结构,如有序数组、链表和B+Tree,来优化数据检索过程,尤其是在大数据量的场景下。了解这些原理有助于我们在设计数据库时做出明智的决策,合理使用索引,以提升系统的整体...

    MySQL 索引最佳实践

    本文将深入探讨MySQL索引的最佳实践,旨在为开发者和数据库管理员提供实用指南。 ### 理解索引的重要性 索引的存在主要是为了加速数据库中的数据访问速度。在没有索引的情况下,数据库必须遍历整个表来查找特定的...

Global site tag (gtag.js) - Google Analytics