MySQL索引
时间:2023-2-27 21:47 作者:wen 分类: MySQL
- 什么事索引
索引(Index)是帮助MySQL高效获取数据的数据结构,可以得到索引的本质:索引是数据结构
索引的目的在于提高查询效率,可以类比字典。
如果要查“mysql”这个单词,我们肯定需要定位到m字母,然后从上往下找到y字母,在找到剩下的sql。如果没有索引,那么你可能需要a----z,如果我想找到Java开头的单词呢?或者Oracle开偶的单词呢?是不是觉得如果没有索引,这个事情根本无法完成?
你可以简单理解为“排好序的快速查找的数据结构”。
索引会影响WHERE和ORDER BY
在数据之外,数据库系统还维护着满足特定查找算法的数据结构,这些数据结构以某种方式引用(指向)数据,这样就可以在这些数据结构上实现高级查找算法。这种数据结构,就是索引。下图就是一种可能的索引方式实例:
为了加快Col2的查找,可以维护一个右边所示的二叉查找树,每个节点分别包含索引键值和一个指向对应数据记录物理地址的指针,这样就可以运用二叉查找在一定的复杂度内获取到相应数据,从而快速的检索出符合条件的记录。
一般来锁索引本身也很大,不可能全部存储在内存中,因此索引往往以索引文件的形式存储在硬盘上
我们平常所说的索引,如果没有特别指明,都是指B数(多路搜索树,并不一定是二叉的)结构组织的索引。其中聚集索引,次要索引,覆盖索引,复合索引,前缀索引,唯一索引默认都是使用B+数索引,统称索引。当然,除了B+树这种类型的索引之外,还有哈稀索引(hash index)等。
-
优势
- 类似大学图书馆建书目索引,提高数据检索的效率,降低数据库的IO成本
- 通过索引列对数据进行排序,降低数据排序的成本,降低了CPU的消耗
-
劣势
- 虽然索引大大提高了查询速度,同时却会降低更新表的速度,如对表进行INSERT、UPDATE和DELETE。因为更新表示,MySQL不仅要保存数据,还要保存一下索引文件每次更新添加了索引列的字段,都会调整因为更新所带来的键值变化后的索引信息。
- 索引只是提高效率的一个因素,如果你的MySQL有大数据的表,就需要花时间研究建立最优秀的索引,或者优化查询语句
-
索引分类
- 单值索引:即一个索引只包含当个列,一个表可以有多个单列索引
- 唯一索引:索引列的值必须唯一,但允许有空值
- 复合索引:即一个索引包含多个类
-
基本语法
- 创建:
CREATE [NUIQUE] INDEX indexName ON mytabel(columnname(length))
ALTER mytabel ADD [UNIQUE] INDEX [indexName] ON (columnname(length))
2.删除:
DROP INDEX [indexName] ON mytable
3.查看:
SHOW INDEX FROM table_name\G
- 创建:
-
索引结构
- BTREE索引:3层的B+数可以表示上百万的数据,如果上百万的数据查找只需要3次IO,性能提高将是巨大的,如果没有索引,每个数据项都要发送一次IO,那么总共需要百万次的IO,显然成本非常非常高。
2.HASH索引
3.full-text全文索引
4.R-Tree索引
- BTREE索引:3层的B+数可以表示上百万的数据,如果上百万的数据查找只需要3次IO,性能提高将是巨大的,如果没有索引,每个数据项都要发送一次IO,那么总共需要百万次的IO,显然成本非常非常高。
-
那些情况需要创建索引
- 主键自动建立唯一索引
- 频繁作为查询条件的字段应该创建索引
- 查询中与其它表关联的字段,外键关系建立索引
- 频繁更新的字段不适合创建索引,因为每次更新不单单更新了记录还会更新索引
- where条件里用不到的字段不要创建索引
- 单键、组合索引的选择(在高并发下倾向创建组合索引)
- 查询中排序的字段,排序字段若通过索引区访问将大大提高排序速度
- 查询中统计或者分组字段
-
那些情况不要创建索引
- 表记录太少
- 经常增删改的表
- 注意,如果某个数据列包含许多重复的内容,为它建立索引就没有太大的实际效果
-
性能分析
MySQL Query OPtimizer -
主要功能:通过计算分析系统中收集到的统计信息,为客户端请求的Query提供他认为最优的执行计划
-
当客户端向MySQL请求一条Query,命令解析器模块完成请求分类,区别出是SELECT并转发给MySQL Query Optimizer时,MySQL Query Optimizer 首先会对整条Query进行优化,处理掉一些常量表达式的预算,直接换算成常量值。并对Query中的查询条件进行简化和转换,如去掉一些无用或显而易见的条件、结构调整等。然后分析Query中的Hint信息(如果有),看显示Hint信息是否可以完全确定该Query的执行计划。如果没有Hint或Hint信息还不足以完全确定执行计划,则会读取所涉及对象的统计信息,根据Query进行写相应的计算分析,然后再得出最后的执行计划。
MySQL常见瓶颈
- CPU:CPU在饱和的时候一般发生在数据装入内存或从磁盘上读取数据时候
- IO:磁盘I/O瓶颈发生在装入数据远大于内存容量的时候
- 服务器硬件的性能瓶颈:top,free,iostat和vmstat来查看系统的性能状态
Explain
使用EXPLAIN关键字可以模拟优化器执行SQL查询语句,从而知道MySQL是如何处理你的SQL语句的,分析你的查询语句或是表结构的性能瓶颈
- 表的读取顺序
- 数据读取操作的操作类型
- 那些索引可以使用
- 那些索引被实际使用
- 表之间的引用
- 每张表有多少行被优化器查询
怎么用: explain + SQL语句
id | select查询的序列号,包含一组数字,表示查询中执行select子句或操作表的顺序三种情况:1.id相同,执行顺序由上至下2.id不同,如果是子查询,id的序号会递增,id值越大优先级越高,越先被执行3.id相同不同,同时存在 |
---|---|
select_type | 1.有哪些:SIMPLE:简单的select查询,查询中不包含子查询或者unionPRIMARY:查询中若包含任何复杂的子部分,最外层查询则被标记为SUBQUERY:在select或where列表中包含了子查询DERIVED:在from列表中包含的子查询被标记为DERIVED(衍生)MySQL会递归执行这些子查询,把结果放在临时表里UNION:若第二个select出现在union之后,则被标记为union;如union包含在from子句的子查询中,外层select将被标记为:DERIVEDUNION RESULT:从union表获取结果的select2.查询的类型,主要是用于区别普通查询,联合查询,子查询等的复杂查询 |
table | 数据关于那张表的 |
type | 显示查询使用了何种类型,ALL:Full Tabel Scan,将遍历全表以找到匹配的行index:Full Index Scan,index于ALL的区别为index类型只遍历索引数。这通常比ALL快,因为索引文件通常比数据文件小。(也就是说虽然all和index都是度全表,但index是从索引中读取的,而all是从硬盘中读的)。range:只检索给定范围的行,使用一个索引来选择行。key列显示使用了那个索引。异步就是在你的where语句中出现了between、<、>、in等的查询这种范围扫描索引扫描比全表扫描要好,因为它只需要开始于索引的某一点,而结束另一点,不用扫描全部索引ref:非唯一性索引扫描,放回匹配某个单独值得所有行。本质上也是一种索引访问,它放回所有匹配某个单独值得行,然而,它可能会找到多个符合条件的行,所以他应该属于查找和扫描的混合体eq_ref:唯一性索引扫码 ,对于每个索引键,表中只有一条记录与之匹配,常见于主键或唯一索引扫描const:如果通过索引一次就找到了,const用户比较primary key或者unique索引。因为只匹配一行数据,所以很快。如果主键置于where列表中,MySQL就能将该查询转换为一个常量。system:表只有一行记录(等于系统表),这是const类型的特例,平时不会出现,这个也可以忽略不计备注:一般来说,得保证查询至少达到range级别,最好能达到ref**从最好到最差依次为:system>const>eq_ref>ref>rangge>index>ALL全部:system>const>eq_ref>fulltext>ref_or_null>index_merge>unique_subquery>index_subquery>rangge>index>ALL** |
possible_keys | 显示可能应用在这张表中的索引,一个或多个。查询涉及到的字段上若存在索引,则该索引将被列出,但不一定被查询实际使用 |
key | 实际使用的索引。如果为NULL,则没有使用所以,查询中若使用了覆盖索引,则该索引仅出现在key列表中 |
key_len | 表示索引中使用的字节数,可通过该列计算查询使用的索引的长度。在不损失精确性的情况下,长度越短越好。key_len显示的值为索引字段的最大可能长度,并非事件长度,即key_len是根据表定义计算而得,不是通过表内检索出的 |
ref | 显示索引的哪一列被使用了,如果可能的话,是一个常量。哪些列或常量被用户查找索引列上的值 |
rows | 根据表统计信息及索引选用情况,大致估算出找到所需的记录所需要读取的行数 |
Extra | 包含不适合在其他列中显示但十分重要的额外信息Using filesort:说明MySQL会对数据使用一个外部的索引排序,而不是按照表内的索引顺序进行读取。MySQL中无法利用索引完成的排序操作称为“文件排序”。Using temporary:使用了临时表保存中间结果,MySQL在对查询结果排序是使用临时表。常见于排序order by 和分组查询group by。Using Index:表示相应的select操作中使用了覆盖索引(Covering index),避免访问了表的数据行,效率不错!如果同时出现using where,表明索引被用来执行索引键值的查找;如果没有同时出现using where,表明索引用来读取数据而非执行查找动作。Using where:表明使用了where过滤Using join buffer:使用了连接缓存impossible where:where子句的值总是false,不能用来获取任何元组。select table optimized away:在没有group by子句的情况下,基于索引优化min/max操作或者对于MyISAM存储引擎优化count(*)操作,不必等到执行阶段再进行计算,查询执行计划生产的阶段即完成优化。distinct:优化distinct操作,在找到第一匹配的元组后即停止找同样值得动作 |
标签: mysql优化