写了三年代码才知道mysql索引这样用

转载请声明出处,尊重版权。

索引的形式

索引是以一定数据格式存储数据的查找路径的数据形式。

这里举个例子:

最典型的就是图书馆找一本书,我们可以通过 “文学” -> “当代” -> “散文” 来缩小我们要找某一本书的范围。那么在这里书本的大分类、时代、小分类,就可以作为找书本的索引。

省份索引例子图-图1

索引的开销

索引很消耗资源,不添加不必要的索引。这里是因为,索引是以树结构来存储的,为了加快检索的效率,减少查询的次数,mysql需要保证这个树的平衡,而这需要花销比较多的资源和时间。

对于写:索引的更新往往是最主要的开销。那么,对于写异常频繁的业务,我们可以考虑删除索引这种非常规的手段。

对于读:索引会占用内存跟硬盘空间。占用内存和硬盘的同时,也加快了检索的速度,典型的以空间换时间的做法。

字符串索引

1.本质上没差别,“AAAA” < “AAAB”(collation)
2.LIKE是特殊的范围查询
3.LIKE “ABC%” 等同于“ABC[LOWEST]”<KEY<“ABC[HIGHEST]”,由于前缀确认,可以使用索引
4.LIKE “%ABC”无法使用索引

Innodb索引

1.数据按主键聚集。也就是说,根据主键可以直接获取数据。
2.主键隐藏添加在所有索引的后面,也就是说 KEY(A)等同于KEY(A,ID)

数据查找

下面语句可以使用索引 (LAST_NAME)

SELECT * FROM EMPLOYEES WHERE LAST_NAME=“Smith”

下面语句可使用索引 (DEPT,LAST_NAME)

SELECT * FROM EMPLOYEES WHERE LAST_NAME=“Smith” AND DEPT=“Accounting”

索引含有多个字段

考虑有索引 (A,B,C),请留意字段的顺序
以下情况能够使用索引

A>5 //用到了(A)
A=5 AND B>6 //用到了(A,B)
A=5 AND B=6 AND C=7 //用到了(A,B,C)
A=5 AND B IN (2,3) AND C>5 //用到了(A,B,C)

以下情况不够使用索引

B>5
B=6 AND C=7

以下情况只能使用到索引的部分字段

//只能用上index(A)
A>5 AND B=2   
//只能用上index(A,B)
A=5 AND B>6 AND C=2 
多字段索引结论

1.索引的匹配规则是左匹配
2.有了(A,B,C),就等于同时拥有了(A)和(A,B)
3.只要索引内,开始用范围查询,后面的索引就失效了。

这里注意:
IN 在 where 中,也属于准确查询,不会使后面索引失效。

索引用于排序

下面语句可以使用索引 (SCORE)

SELECT * FROM PLAYERS ORDER BY SCORE DESC LIMIT 10

下面语句可使用索引 (COUNTRY, SCORE)

SELECT * FROM PLAYERS WHERE COUNTRY=“US” ORDER BY SCORE DESC LIMIT 10

多字段索引与排序

考虑有索引(A, B),留意字段顺序
以下情况能够使用索引

ORDER BY A  
A=5 ORDER BY B
ORDER BY A DESC, B DESC
A>5 ORDER BY A

以下情况不能使用索引用于排序

ORDER BY B //第二个字段上面排序
A>5 ORDER BY B //第一个字段是范围查询
A IN(1,2) ORDER BY B //第一个字段是IN范围查询
ORDER BY A ASC, B DESC //排序字段的顺序不一致

最左匹配

对于一个索引来说,里面的索引是有先后顺序的。
例如:索引(A,B,C)

系统建立索引时,mysql会这么做:
首先,会根据 A 的值,构建索引,
在相同 A 索引下,建立 B 的索引,
在 B 相同的索引下,建立 C 的单独索引。
再多的字段也是如此类推。

那么

索引组内,当前索引无法在前缀不确定的情况下使用当前索引。

这也就是下面这几个语句无法使用索引或无法使用完整索引的原因:

A LIKE '%aaa' //前缀不确定,无法使用索引
A>5 AND B=2   //前缀A不确定,只能用到 (A),无法用(A,B)
A IN(1,2) ORDER BY B  //只能用到 (A),无法用(A,B)

那为什么“A=5 AND B IN (2,3) AND C>5” 可以用到(A,B,C)呢?

实际上,mysql优化器在这里做了优化,
查询的时候拆分了两次检查索引:

A=5 AND B=2 AND C>5
A=5 AND B=3 AND C>5

那么,在IN的范围不太大的情况下,可以在很大程度上优化查询速度。

MySQL是如何选择索引的?

1.每次查询动态选择

2.估算走索引需要查询的行数

3.根据“Cardinality”的状态。作为重要参考标准。大致原理是:索引重叠的程度,重叠程度越低,这个基数越大,优化器优先选取。

show index from table 执行结果

Cardinality - 图2

此图告诉我们:
使用索引查找速度对比为:customer_id > i_gid > company_id

索引策略

1.给最频繁的语句加索引 --要整体来看,而不是一条一条语句添加
2.尽可能扩展索引,而不是新增索引
3.WHERE条件跟JOIN都能用上索引是最好的,
4.新增索引后要验证索引是否生效,是否对能提升性能

例子:
SELECT * FROM TBL WHERE A=5 AND B=6
SELECT * FROM TBL WHERE A>5 AND B=6
应该新增索引(B,A)

优化技巧

技巧1:范围查询改为枚举类型

考虑索引(A,B),A 类型为 int

//只能用到(A)索引
SELECT * FROM TBL WHERE A BETWEEN 2 AND 4 AND B=5

改为

//可以用到整个索引
SELECT * FROM TBL WHERE A IN (2,3,4) AND B=5 

技巧2:增加假的条件

考虑索引KEY (GENDER,CITY),GENDER性别为可枚举数据

//无法使用索引
SELECT * FROM PEOPLE WHERE CITY=“NEW YORK”

改为

//能用索引
SELECT * FROM PEOPLE WHERE GENDER IN (“M”,”F”) AND CITY=“NEW 

对于Gender, Status, Boolean 类型非常有效

技巧3:Unionizing Filesort

考虑索引KEY(A,B)

//无法使用索引
SELECT * FROM TBL WHERE A IN (1,2) ORDER BY B LIMIT 5;

改为

//能使用索引用作排序,没有file_sort
(SELECT * FROM TBL WHERE A=1 ORDER BY B LIMIT 5) UNION ALL
(SELECT * FROM TBL WHERE A=2 ORDER BY B LIMIT 5) ORDER BY B LIMIT 5; 

Bug: order by limit

考虑索引KEY(A,B),KEY(C)

select * from tb where a=1 and b =2 order by c  limit 10

有时候不能正确的选择索引,
会指定使用索引(C),
先排序后筛选,导致全面检索。
这时候需要指定索引或者改为

select * from tb where a=1 and b =2 order by c+0  limit 10

EXPLAIN

关于EXPLAIN语句,可以帮助我们去了解一条语句的执行效率和所用到的索引,由于太复杂,可以去自行看官方的文档。有空可以继续写一篇文章去帮助大家理解,也帮助自己总结。
EXPLAIN命令文档

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,012评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,628评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,653评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,485评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,574评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,590评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,596评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,340评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,794评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,102评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,276评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,940评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,583评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,201评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,441评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,173评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,136评论 2 352

推荐阅读更多精彩内容