数据库索引的作用及原理

数据库索引是为了增加查询速度而对表字段附加的一种标识。很多人机械的理解索引的概念认为增加索引只有好处没有坏处。其实远不是那样的这里将其介绍尽量详细些。


首先明白为什么索引会增加速度,DB在执行一条Sql语句的时候默认的方式是根据搜索条件进行全表扫描遇到匹配条件的就加入搜索结果集合。如果我们对某一字段增加索引查询时就会先去索引列表中一次定位到特定值的行数大大减少遍历匹配的行数所以能明显增加查询的速度。那么在任何时候都应该加索引么这里有几个反例:1、如果每次都需要取到所有表记录无论如何都必须进行全表扫描了那么是否加索引也没有意义了。2、对非唯一的字段例如性别这种大量重复值的字段增加索引也没有什么意义。3、对于记录比较少的表增加索引不会带来速度的优化反而浪费了存储空间因为索引是需要存储空间的而且有个致命缺点是对于update/insert/delete的每次执行字段的索引都必须重新计算更新。


那么在什么时候适合加上索引呢看这样一个例子这里有一条sql语句


SELECT c.companyID, c.companyName FROM Companies c, User u 

WHERE c.companyID = u.fk_companyID AND c.numEmployees >= 0 

AND c.companyName LIKE '%i%' 

AND u.groupID IN (SELECT g.groupID FROM Groups g WHERE g.groupLabel = 'Executive') 


这条语句涉及3个表的联接并且包括了许多搜索条件比如大小比较,Like匹配等。在没有索引的情况下Mysql需要执行的扫描行数是77721876行。而我们通过在companyIDgroupLabel两个字段上加上索引之后扫描的行数只需要134行。在Mysql中可以通过Explain Select来查看扫描次数。可以看出来在这种联表和复杂搜索条件的情况下索引带来的性能提升远比它所占据的磁盘空间要重要得多。


那么索引是如何实现的呢大多数DB厂商实现索引都是基于一种数据结构——B树。因为B树的特点就是适合在磁盘等直接存储设备上组织动态查找表。B树的定义是这样的一棵m(m>=3)阶的B树是满足下列条件的m叉树


1、每个结点包括如下作用域(j, p0, k1, p1, k2, p2, ... ki, pi) 其中j是关键字个数,p是孩子指针


2、所有叶子结点在同一层上层数等于树高h


3、每个非根结点包含的关键字个数满足[m/2-1]<=j<=m-1


4、若树非空则根至少有1个关键字若根非叶子则至少有2棵子树至多有m棵子树


在数据量比较大的情况下这样的结构可以大大增加查询速度。然而有另外一种数据结构查询的虚度比B树更快——散列表。Hash表的定义是这样的设所有可能出现的关键字集合为u,实际发生存储的关键字记为k,|k||u|小很多。散列方法是通过散列函数hu映射到表T[0,m-1]的下标上这样u中的关键字为变量h为函数运算结果即为相应结点的存储地址。从而达到可以在o(1)的时间内完成查找。


然而散列表有一个缺陷那就是散列冲突即两个关键字通过散列函数计算出了相同的结果。设mn分别表示散列表的长度和填满的结点数,n/m为散列表的填装因子因子越大表示散列冲突的机会越大。 因为有这样的缺陷所以数据库不会使用散列表来做为索引的默认实现,Mysql宣称会根据执行查询格式尝试将基于磁盘的B树索引转变为和合适的散列索引以追求进一步提高搜索速度。





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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值