知识点二十八:索引

前言

在实际的软件开发中,业务纷繁复杂,功能千变万化,但是,万变不离其宗,如果抛开这些业务和功能的外壳,其实它们的本质都可以抽象为“对数据的存储和计算”。对应到数据结构和算法中,那“存储”需要的就是数据结构,“计算”需要的就是算法。

对于存储的需求,功能上无外乎增删改查。这其实并不复杂。但是,一旦存储的数据很多,那性能就成了这些系统要关注的重点,特别是在一些跟存储相关的基础系统(比如 MySQL 数据库、分布式文件系统等)、中间件(比如消息中间件 RocketMQ 等)中。“如何节省存储空间、如何提高数据增删改查的执行效率”,这样的问题就成了设计的重点。而这些系统的实现,都离不开一个东西,那就是索引。不夸张地说,索引设计得好坏,直接决定了这些系统是否优秀。毫不夸张地说,索引设计得好坏,将直接决定这些系统是否优秀。

索引

索引这个概念,可以类比书籍的目录来理解。如果没有目录,那当我们想要查找书中某个知识点的时候,就要一页一页翻。但是,如果通过目录,我们就可以快速定位到相关知识点的页数,查找的速度会有质的提高。如果用一句话来描述“索引”的作用,那就是:索引(index)是用来辅助查找,用计算机专业术语就叫做 寻址(Addressing)。例如,数组中的下标就是index。

理解了索引的概念后,接下来,我们就分析一下,在设计索引的过程中,一般需要考虑到的一些因素,换句话说就是,我们该如何定义清楚索引的需求呢?对于系统设计需求,我们一般可以从功能性需求和非功能性需求两方面来分析,这个问题也不例外。

1.功能性需求

对于功能性需求需要考虑的点,大致概括成下面这几点。实际上,不同的场景,不同的原始数据,对于索引的需求也会千差万别。这里也只是列举了一些比较有共性的需求。

  • 数据是结构化数据还是非结构化数据?
    要构建索引的原始数据,类型有很多。主要分为两类,一类是结构化数据,比如,MySQL 中的数据;另一类是非结构化数据,比如搜索引擎中网页。对于非结构化数据,我们一般需要做预处理,提取出查询关键词,对关键词构建索引。
  • 数据是静态数据还是动态数据?
    如果原始数据是一组静态数据,也就是说,不会有数据的增加、删除、更新操作,所以,我们在构建索引的时候,只需要考虑查询效率就可以了。这样,索引的构建就相对简单些。不过,大部分情况下,我们都是对动态数据构建索引,也就是说,我们不仅要考虑到索引的查询效率,还需要在原始数据更新的同时,动态地更新索引。支持动态数据集合的索引,设计起来相对也要更加复杂些。
  • 索引存储在内存还是硬盘?
    如果索引存储在内存中,那查询的速度肯定要比存储在磁盘中的高很多。但是,如果在原始数据量很大的情况下,对应的索引可能也会很大,这个时候,因为内存有限,我们可能就不得不将索引存储在磁盘中了。实际上,还有第三种情况,那就是一部分存储在内存,一部分存储在磁盘,这样就可以兼顾内存消耗和查询效率。
  • 单值查找还是区间查找?
    所谓单值查找,也就是查询等于某个值的数据,这种查询需求最常见。而所谓的区间查找,就是查找处于某个区间内的所有数据。实际上,不同的应用场景,查询的需求会多种多样,比如 MySQL 数据库中的查询需求就不限于上述两种。
  • 单关键词查找还是多关键词组合查找?
    比如,搜索引擎中构建的索引,既要支持一个关键词的查找,比如“数据结构”,也要支持组合关键词的查找,比如“数据结构 AND 算法”。对于单关键词的查找,索引构建起来相对简单些。对于多关键词查询来说,要分多种情况。像 MySQL 这种结构化数据的查询需求,我们可以针对多个关键词的组合建立索引;而对于像搜索引擎这样的非结构数据的查询需求,我们可以针对单个关键词构建索引,然后通过集合操作,比如求并集、求交集等,计算出多个关键词组合的查询结果。

2.非功能性需求

对于非功能性需求,我们着重考虑性能方面的需求。具体来说,性能方面的需求,我们主要考察时间和空间两方面,也就是执行效率和存储空间。在执行效率方面,我们希望通过索引,查询数据的效率尽可能的高;在存储空间方面,我们希望索引不要消耗太多的内存空间。

  • 不管是存储在内存中还是磁盘中,索引对存储空间的消耗都不能过大。如果存储在内存中,索引对占用存储空间的限制就会非常苛刻。毕竟内存空间非常有限,例如一个中间件启动后就会占用几个 GB 的内存,开发者显然是无法接受的。如果存储在硬盘中,那索引对占用存储空间的限制,稍微会放宽一些。但是,我们也不能掉以轻心。因为,有时候,索引对存储空间的消耗甚至会超过原始数据。
  • 在考虑索引查询效率的同时,我们还要考虑索引的维护成本。索引的目的是提高查询效率,但是,基于动态数据集合构建的索引,我们还要考虑到索引的维护成本。因为在原始数据动态增删改的同时,我们也需要动态地更新索引。而索引的更新势必会影响到增删改操作的性能。

构建索引常用的数据结构有哪些?

刚刚我们从很宏观的角度,总结了在索引设计的过程中,需要考虑的一些共性因素。现在,我们再来看,对于不同需求的索引结构,底层一般使用哪种数据结构。实际上,常用来构建索引的数据结构,就是我们之前讲过的几种支持对动态数据集合快速查询、插入等操作的数据结构。比如,散列表、红黑树、跳表、B+ 树。除此之外,位图、布隆过滤器可以作为辅助索引,有序数组可以用来对静态数据构建索引。

  • 散列表增删改查操作的性能非常好,时间复杂度是 O(1)。一些键值数据库,比如 Redis、Memcache,就是使用散列表来构建索引的。这类索引,一般都构建在内存中
  • 红黑树作为一种常用的平衡二叉查找树,数据插入、删除、查找的时间复杂度都是 O(logn),也非常适合用来构建存储在内存中的索引。比如,Ext 文件系统中对磁盘块的索引,用的就是红黑树。
  • 相比红黑树,B+ 树更加适合构建存储在磁盘中的索引。B+ 树是一棵多叉树,所以,对相同个数的数据构建索引,B+ 树的高度要低于红黑树。当借助索引查询数据的时候,读取 B+ 树索引需要的磁盘 IO 次数会更少。所以,大部分关系型数据库的索引,比如 MySQL、Oracle等,都是用 B+ 树来实现的。
  • 跳表也支持快速添加、删除、查找数据。而且,跳表中通过灵活调整索引结点个数和数据个数之间的比例,可以很好地平衡索引对内存的消耗及其查询效率。比如,Redis 中的有序集合,就是用跳表来构建的。
  • 位图和布隆过滤器这两个数据结构,也可以用于索引中,辅助存储在磁盘中的索引,加速数据查找的效率。我们知道,布隆过滤器有一定的判错率。但是,我们可以规避它的短处,发挥它的长处。尽管对于判定存在的数据,有可能并不存在,但是对于判定不存在的数据,那肯定就不存在。而且,布隆过滤器还有一个更大的特点,那就是内存占用非常少。我们可以针对数据,构建一个布隆过滤器,并且存储在内存中。当要查询数据的时候,我们可以先通过布隆过滤器,判定是否存在。如果通过布隆过滤器判定数据不存在,那我们就没有必要读取磁盘中的索引了。对于数据不存在的情况,数据查询就更加快速了。
  • 有序数组也可以被作为索引。如果原始数据是静态的,也就是不会有插入、删除、更新的操作,那我们可以把查询数据用的关键词抽取出来,组织成有序数组,然后利用二分查找算法来快速查找数据。

小结

一、索引
1.概念:在一些跟存储相关的基础系统(比如 MySQL 数据库、分布式文件系统等)、中间件(比如消息中间件 RocketMQ 等)中,都离不开一个东西,那就是索引。索引设计得好坏,直接决定了这些系统是否优秀。总的来说,索引就是用来辅助查找,从而节省存储空间,提高数据增删改查的执行效率。
2.索引的需求定义
(1)功能性需求:数据是格式化数据还是非格式化数据、数据是静态数据还是动态数据、索引存储在内存还是硬盘、单值查找还是区间查找、单关键词查找还是多关键词组合查找……
(2)非功能性需求:不管是存储在内存中还是磁盘中,索引对存储空间的消耗不能过大;在考虑索引查询效率的同时,我们还要考虑索引的维护成本。
3.构建索引常用的数据结构:散列表、红黑树、跳表、B+ 树。除此之外,位图、布隆过滤器可以作为辅助索引,有序数组可以用来对静态数据构建索引。

二、引申
通过索引这个非常常用的技术方案,向你展示了散列表、红黑树、跳表、位图、布隆过滤器、有序数组这些数据结构的应用场景。实际上,架构设计离不开数据结构和算法。要想成长为一个优秀的业务架构师、基础架构师,数据结构和算法的根基一定要打稳。因为,那些看似很惊艳的架构设计思路,实际上,都是来自最常用的数据结构和算法。

参考

《数据结构与算法之美》
王争
前Google工程师

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值