深入MySQL(三):MySQL的索引的应用
在MySQL的优化中,索引的作用绝对算是一个大头,很多时候索引使用得当可以使得一个查询的效率提高几个数量级,同时它还具有自动排序等功能。所以如果是深入MySQL,那么索引绝对是其中重要的一部分。
MySQL中的索引
MySQL中的索引按类型分大致可以分为5类:
- 主键索引
- 普通索引
- 唯一索引
- 文本索引
- 组合索引
主键索引
定义:唯一且不为空,简称主键,表中可以没有主键,但是最多只能有一个;
在Innodb中,会默认将主键索引设置为聚簇索引;
一般而言,主键索引的建立符合以下的原则:
- 与业务无关的自增属性作为主键;
- 不选取可以动态变化的属性,例如时间戳;
- 字段值不进行更新;
用户量较小的情况下,一般用自增序列即可满足条件,但是在一些情况下,主键的选择就不只是简单的自增属性了,后续会专门出一篇博客重新讲述分布式唯一ID的生成算法。
普通索引(NORMAL)
定义:设置单个字段作为索引,没有任何限制,是我们使用最多的索引;
普通索引的查询,在innodb中其是非聚簇索引,索引它存在一个回表的过程,所以后期有个优化便是使用覆盖索引;
唯一索引(UNIQUE)
定义:设置单个字段作为索引,并且该索引不会重复,可以为null;
主要是对该索引字段进行唯一性限制。
组合索引
定义:设置多个字段作为索引;
将多个字段作为一个索引,索引选取的字段的顺序很重要,遵循最左匹配原则。
以上是按照索引的使用类型来划分的;
按底层数据结构分可以分为2类:
- B-Tree索引
- hash索引
Innodb中默认使用的便是B-Tree索引,其底层数据结构是B+树,对以下类型的查询有效:
- 全值匹配:指的是和索引中的所有列进行匹配,即例如WHERE name = 'hyx' AND age = 18的查询;
- 匹配最左前缀:即组合索引中的使用前面的字段进行匹配;
- 匹配列前缀:LIKE的用法也是复合最左匹配原则的;
- 匹配范围值:B-Tree索引会按照大小将数据排序存放;
- 精确匹配某一列并范围匹配另外一列;
- 只访问索引的查询;
但是B-Tree索引也有缺点:
- 如果不符合最左匹配原则,那么无法进行匹配;
- 不能跳过索引中的列;我的理解是,假如我WHERE name = 'hyx' AND age = 18,在name字段上有一个普通索引,数据表中有很多个name为hyx的用户,这个时候不能直接跳过中间age != 18的列,而是要一个一个取出来,然后回表去查age,最后在判断是否为18,相当于必须扫描这些不正确的数据行。
hash索引使用的底层数据结构为Hash表,等值匹配速度很快,但是范围遍历效率低。 - 一旦使用范围查询,那么其右边的所有列都无法走索引;我认为这也是最左匹配原则的体现;
大家可以看到,B-Tree索引的很多运用场景都与最左匹配原则有关,所以大家写查询的时候一定记住思考最左匹配原则,同时通过EXPLAIN来查看语句的执行计划来优化索引;
Hash索引的特点:
- 哈希索引只包含哈希值和行指针,而不存储字段值,所以无法使用覆盖索引的做法避免读取行。
- 不会自动进行排序;
- 不支持索引列匹配,比如一个数据列(A,B)上建立索引,它始终是使用索引列的全部内容进行计算哈希值的,所以如果查询只含有数据列A,无法使用索引;
- 只支持等值比较;
- 访问速度极快,除非hash冲突很多;
- hash冲突很多的时候,维护的代价也会提高。
Hash索引的话,只适用于某些特定的场合。
建立高性能索引的策略
按照经验来说,其实我们使用WHERE的字段,就可以建立索引;但是作为技术人员,我们应该严谨的研究什么时候添加索引。我们这里暂时都只对B-Tree索引进行讨论,先来看B-Tree索引带来的好处:
- 索引大大减少了扫描的行数;
- 索引可以避免排序和临时表;
- 索引可以将随机IO变为顺序IO;
独立的列
独立的列主要代表两个方面:
- 索引列不能是表达式的一部分;
- 索引列不能是函数的参数;
其实我觉得很好理解,因为以上的两种情况,都要对列进行计算,相当于就不是原来的列了,所以就无法使用索引;
使用前缀索引
我们都知道,现实中是存在例如邮箱等的具有相同后缀的数据的,所以我们肯定指定使用其前缀索引来减小和加快索引的查找,但是会降低选择性;
选择性是不重复的列 / 总数据列数的值,最好为1,我们可以使用COUNT()函数来计算其选择性,来选取与原来的选择性相差不大的情况;
使用覆盖索引避免回表
我们知道,如果在一个字段上面建立一个索引,那么索引的节点其实是存储了值,主键的,获取主键之后,再需要去主键所在的B+树进行查找,才能查找到具体数据。
而如果使用了覆盖索引,例如我们建立一个组合索引“email,phone”,当我们利用WHERE email = '843497509@qq.com'查找phone的时候,其实我们已经获取了phone的数据,所以MySQL不会进行回表操作,而是直接返回结果集。
利用索引进行排序
MySQL的排序是一个很麻烦很复杂很耗性能的一个过程,甚至会进行大量的磁盘IO,所以如果某一个字段需要经常性的排序,在B+树中,会自动根据B+树的节点值进行排序,所以我们就会省略这个字段的排序过程。
深入MySQL(三):MySQL的索引的应用的更多相关文章
- MySQL(三)
MYSQL(三) 上一章给大家说的是数据库的视图,存储过程等等操作,这章主要讲索引,以及索引注意事项,如果想看前面的文章,url如下: MYSQL入门全套(第一部) MYSQL入门全套(第二部) 索引 ...
- mysql btree与hash索引的适用场景和限制
btree索引: 如果没有特别指明类型,多半说的就是btree索引,它使用btree数据结构来存储数据,大多数mysql引擎都支持这种索引,archive引擎是一个例外,5.1之前这个引擎不支持任何索 ...
- 【MySQL】MySQL索引背后的之使用策略及优化【转】
转自:http://database.ctocio.com.cn/353/11664853.shtml 另外很不错的对于索引及索引优化的文章: http://www.cnblogs.com/magia ...
- MySQL源码:索引相关的数据结构
http://www.orczhou.com/index.php/2012/11/mysql-source-code-data-structure-about-index/ 本文将尝试介绍MySQL索 ...
- mysql建立、删除索引及使用
同步发布:http://www.yuanrengu.com/index.php/2017-01-13.html 一.索引的作用 一般的应用系统,读写比例在10:1左右,而且插入操作和一般的更新操作很少 ...
- MySQL中的联合索引学习教程
MySQL中的联合索引学习教程 这篇文章主要介绍了MySQL中的联合索引学习教程,其中谈到了联合索引对排序的优化等知识点,需要的朋友可以参考下 联合索引又叫复合索引.对于复合索引:Mysql从左到 ...
- MySQL性能调优——索引详解与索引的优化
--索引优化,可以说是数据库相关优化.理解尤其是查询优化中最常用的优化手段之一.所以,只有深入索引的实现原理.存储方式.不同索引间区别,才能设计或使用最优的索引,最大幅度的提升查询效率! 一.BTre ...
- SQL优化 MySQL版 - B树索引详讲
SQL优化 MySQL版 - -B树索引详讲 作者:Stanley 罗昊 [转载请注明出处和署名,谢谢!] 为什么要进行SQL优化呢?很显然,当我们去写sql语句时: 1会发现性能低 2.执行时间太 ...
- mysql 最左匹配 联合索引
mysql建立多列索引(联合索引)有最左前缀的原则,即最左优先,如: 如果有一个2列的索引(col1,col2),则已经对(col1).(col1,col2)上建立了索引:如果有一个3列索引(col1 ...
- 【面试】MySQL的事务和索引
MySQL事务 MySQL事务主要用于处理操作量大,复杂度高的数据. 比如说,在人员管理系统中,你删除一个人员,你既需要删除人员的基本资料,也要删除和该人员相关的信息,如信箱,文章等等,这些数据库操作 ...
随机推荐
- Struts-S2-045漏洞利用
最近也是在看Struts2的漏洞,这里与大家共同探讨一下,本次我复现的是s2-045这个编号的漏洞 漏洞介绍 Apache Struts 2被曝存在远程命令执行漏洞,漏洞编号S2-045,CVE编号C ...
- Genymotion模拟器配置与使用
1.https://www.genymotion.com网站注册模拟器账号,并按照指引激活账号 此过程中提示商业用还是个人使用,商业使用需要有注册码,个人使用不需要注册码,我选择的是个人使用. 2.激 ...
- Windows 和 Ubuntu 的网络能互相 ping 通之后,linux无法上网原因:①路由没设置好,②DNS 没设置好
确保 Windows 和 Ubuntu 的网络能互相 ping 通之后,如果 Ubuntu 无法上网,原因通常有 2 个:路由没设置好,DNS 没设置好. 如果执行以下命令不成功,表示路由没设置好: ...
- Choregraphe 2.8.6.23虚拟Nao机器人Socket is not connected
Traceback (most recent call last): File "c:/Users/fengmao/OneDrive - University of Wollongong/J ...
- Natasha 4.0 探索之路系列(四) 模板 API
Natasha 模板 Natasha 在编译单元的基础上进行了封装整理, 并提供了多种模板帮助开发者构建功能. 使用此篇的 API 前提是您对 C# 非常熟悉, 对系统的一些类型足够了解. 据此 Na ...
- Servlet-base标签的作用(相对路径和绝对路径)
Web中的相对路径和绝对路径 在javaWeb中,路径分为相对路径和绝对路径: 相对路径: . 表示当前目录 .. 表示上一级目录 资源名 表示当前目录/资源名 绝对路径: http://ip ...
- 使用 C# 开发 Kubernetes 组件,获取集群资源信息
写什么呢 前段时间使用 C# 写了个项目,使用 Kubernetes API Server,获取信息以及监控 Kubernetes 资源,然后结合 Neting 做 API 网关. 体验地址 http ...
- python类和函数
#/usr/bin/python #coding=utf-8 class Car(): def __init__(self,name,year): self.name = name self.year ...
- 使用VSCode在本地电脑上对树莓派远程开发
目的及原理 有时身边没有额外的显示器和键盘,或者有时树莓派在另一个屋子连接着路由器,那么当我们想在树莓派上做开发时就可以使用VS Code的远程开发能力.下面一张图显而易见地说明了远程开发的工作原理( ...
- kibana 对es的简单操作。
一.查询和查看. #1.查询所有的数据 GET _search { "query": { "match_all":{} } } #2. 查看ES集群的健康状态 ...