MySQL索引调优实战

12 篇文章 0 订阅

MySQL索引调优实战

通过50w条数据的表来进行调优实战,MySQL使用阿里云丐版服务器,性能较慢,获取表数据请私信我

大致的表结构
请添加图片描述

单表查询优化及索引失效情况

全值匹配情况

利用索引进行全值匹配效率更高

EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE age=30;
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE age=30 AND classId=4;
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE age=30 AND classId=4 AND NAME = 'abcd';

无索引情况下:
都是全表扫描,耗时达到一秒,条件越多耗时越长

建立索引:

CREATE INDEX idx_age ON student(age);
CREATE INDEX idx_age_classid ON student(age,classId);
CREATE INDEX idx_age_classid_name ON student(age,classId,NAME);

可以看到消耗时间有所降低,简单测试

最左前缀法则

使用上述创建的3个索引情况解析,最左前缀法则,只能依次向后使用索引字段,一旦跳过了,后边的字段就无法使用索引

# 可以使用age索引
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE student.age=30 AND student.name = 'abcd' ;
# 只能全表扫描,因为没有以classid开头的索引
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE student.classid=1 AND student.name = 'abcd';
# 可以使用三个字段的索引,和SQL编写顺序无关
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE classid=4 AND student.age=30 AND student.name = 'abcd'; 

主键插入的顺序

如果不按照依次递增顺序插入数据,就可能导致页分裂

运算,函数,类型转换导致索引失效

在WHERE字段进行这些操作都会导致全表扫描,因为需要对每行数据都进行这些操作之后才能进行判断是否符合条件

# 创建name字段索引
CREATE INDEX idx_name ON student(NAME);
# 可以使用索引,索引的范围查询,扫描索引的一部分
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE student.name LIKE 'abc%';
# WHERE字段使用函数,导致索引失效
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE LEFT(student.name,3) = 'abc'; 
# WHERE字段运算导致索引失效
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE age + 1 = 30; 
# 类型转换导致索引失效,因为Name是字符串,转换成了数字
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE NAME = 123; 

总结:避免在WHERE后进行运算,函数,类型转换操作

范围条件右边的列索引失效

当字段使用> < <> between范围查询时,索引右边字段就用不上索引了,将>,< 改成<=,>=后可以用上后续索引

# 创建三字段联合索引
CREATE INDEX idx_age_classId_name ON student(age,classId,NAME);
# 中间字段使用>范围查询,导致只有前两个字段使用到了索引(通过key_len长度来发现的)
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE student.age=30 AND student.classId>20 AND student.name = 'abc' ;

不等于(!= 或者<>)索引失效

两种情况都会使索引失效

CREATE INDEX idx_name ON student(NAME);
# ALL
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE student.name <> 'abc' ;
# ALL
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE student.name != 'abc' ;

is null可以使用索引,is not null无法使用索引

# ref
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE age IS NULL;
# ALL 
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE age IS NOT NULL; 

like以通配符%开头索引失效

# range
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE NAME LIKE 'ab%'; 
# ALL
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE NAME LIKE '%ab%';

OR 前后存在非索引的列,索引失效

索引列用索引,非索引列全表扫描,加起来还不如直接全表扫描,所以索引失效

# 对age做索引
CREATE INDEX idx_age ON student(age);
# OR后的classid非索引,导致age也无法使用索引
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE age = 10 OR classid = 100;
# 创建classid索引后,发现上边sql可以使用了索引
CREATE INDEX idx_cid ON student(classid);

字符集不统一导致索引失效

因为不同意就涉及了类型转换,导致索引失效

多表查询优化

解释外连接和内连接索引优化

左外连接

# 添加被连接表的索引
CREATE INDEX Y ON book(card);
# 结果连接表ALL,被链接表ref
EXPLAIN SELECT SQL_NO_CACHE * FROM `type` LEFT JOIN book ON type.card = book.card;
# 两张表都上索引
CREATE INDEX X ON `type`(card);
# 结果连接表index,被链接表ref
EXPLAIN SELECT SQL_NO_CACHE * FROM `type` LEFT JOIN book ON type.card = book.card;

内连接

# 查询优化器决定谁来做驱动表,谁做被驱动表
EXPLAIN SELECT SQL_NO_CACHE * FROM `type` INNER JOIN book ON type.card = book.card;

结论:

  • 对于内连接来说,查询优化器可以决定谁作为驱动表,谁作为被驱动表出现的
  • 对于内连接来讲,如果表的连接条件中只能有一个字段有索引,则有索引的字段所在的表会被作为被驱动表出现
  • 对于内连接来说,在两个表的连接条件都存在索引的情况下,会选择小表作为驱动表。“小表驱动大表”

JOIN语句原理

JOIN方式连接多个表,本质就是各个表之间数据的循环匹配,MySQL5.5之前,MySQL只支持一种表间关联方式,就是嵌套循环,如果关联表的数据量很大,则JOIN关联的执行时间会非常长,5.5之后,MySQL引入BNLJ散算法来优化

驱动表:在Explain时上边的表
被驱动表:在Explain时下边的表

简单循环嵌套:

  • 从驱动表中取一条数据,遍历非驱动表进行连接
  • 直到驱动表遍历结束
  • 所以小表来驱动大表读取次数最少
    索引嵌套循环连接:
  • 对被索引表加索引
  • 内层循环不需要全表扫描,提高性能
    块嵌套循环连接:
  • 不再逐条获取驱动表,而是一块一块获取,引入了join buffer,将驱动表join相关的部分数据列缓存到join buffer中,然后全表扫描非驱动表
    HASH JOIN
  • 在8.0.20后废弃了块嵌套循环连接,默认使用hash join
  • 使用两个表中较小的表利用Join key在内存中建立散列表,然后扫描较大的表并探测散列表,找出与hash表匹配的行
    总结:
  • 用小的结果集去驱动大的结果集
  • 使用STRAIGHT_JOIN来强制驱动表和非驱动表,不让优化器优化

子查询优化

MySQL4.1之后支持子查询,可以进行SELECT语句的嵌套查询,即一个SELECT查询结果作为另一个SELECT语句的条件

子查询效率不高的原因:

  • 执行子查询时,MySQL需要为内存查询语句的查询结果建立一个临时表,然后外层查询语句从临时表中查询记录,查询完毕后再撤销,这样就会造成过多的IO和CPU资源
  • 子查询结果集存储的临时表,无法建立索引,不会使用索引
  • 子查询建议拆成多表查询

排序优化

MySQL中,支持两种排序方式,分别是FileSortIndex排序

  • index排序中,索引可以保证数据的有序性,不需要再排序
  • FileSort排序一般再内存中进行排序,占用CPU较多,如果等待排序结果较大,会产生临时文件IO到磁盘进行排序,效率低

优化建议:

  • SQL中在ORDER BY字句中使用索引,目的是避免FileSort**(不一定FileSort就比Index慢,由优化器去选择)**
  • 尽量使用Index完成ORDER BY排序,如果WHERE和ORDER BY后面是相同的列就使用单索引,如果不同就使用联合索引
  • 无法使用Index时,需要对FileSort进行调优
# 创建联合索引
CREATE  INDEX idx_age_classid_name ON student (age,classid,NAME);
# 对所有数据进行排序时,索引失效(主要是因为需要回表)
EXPLAIN  SELECT SQL_NO_CACHE * FROM student ORDER BY age,classid; 
# 这时候就可以用上索引了(覆盖索引,无须回表)
EXPLAIN  SELECT SQL_NO_CACHE age,classid,name,id FROM student ORDER BY age,classid; 
# 数据量较少,即使回表也比FileSort快,使用了索引
EXPLAIN  SELECT SQL_NO_CACHE * FROM student ORDER BY age,classid LIMIT 10; 

order by时顺序错误,索引失效

主要看最左前缀法则

#以下哪些索引失效?
# 不用
EXPLAIN  SELECT * FROM student ORDER BY classid LIMIT 10;
# 不用
EXPLAIN  SELECT * FROM student ORDER BY classid,NAME LIMIT 10;  
# 用
EXPLAIN  SELECT * FROM student ORDER BY age,classid,stuno LIMIT 10; 
# 用
EXPLAIN  SELECT * FROM student ORDER BY age,classid LIMIT 10;
# 用
EXPLAIN  SELECT * FROM student ORDER BY age LIMIT 10;

order by时规则不一致, 索引失效 (顺序错,不索引;方向反,不索引)

看排序方向

# 不用,age降序了
EXPLAIN  SELECT * FROM student ORDER BY age DESC, classid ASC LIMIT 10;
# 不用,没给到age
EXPLAIN  SELECT * FROM student ORDER BY classid DESC, NAME DESC LIMIT 10;
# 不用,看来有降序就不行了
EXPLAIN  SELECT * FROM student ORDER BY age ASC,classid DESC LIMIT 10; 
# 都是降序,就可以用索引了,因为可以反着遍历叶子节点
EXPLAIN  SELECT * FROM student ORDER BY age DESC, classid DESC LIMIT 10;

无过滤,不索引

# 只用了age
EXPLAIN  SELECT * FROM student WHERE age=45 ORDER BY classid;
# 只用了age
EXPLAIN  SELECT * FROM student WHERE  age=45 ORDER BY classid,NAME; 
# 不用
EXPLAIN  SELECT * FROM student WHERE  classid=45 ORDER BY age;
# 会用,先用age排序,再找classid=45
EXPLAIN  SELECT * FROM student WHERE  classid=45 ORDER BY age LIMIT 10;

排序实战

# 默认情况使用FileSort     1.417s
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE age = 30 AND stuno <101000 ORDER BY NAME ;
# 构建索引防止FileSort        1.118s
CREATE INDEX idx_age_name ON student(age,NAME);
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE age = 30 AND stuno <101000 ORDER BY NAME ;
# 建立WHERE查询条件的索引
CREATE INDEX idx_age_stuno_name ON student(age,stuno,NAME);
# 因为这个索引效率很高了,索引Name字段进行FileSort就可以,用不用Name字段取决于过滤来的数据量大小
EXPLAIN SELECT SQL_NO_CACHE * FROM student WHERE age = 30 AND stuno <101000 ORDER BY NAME ;

这里总结:
当出现范围条件和group by字段二选一时,优先观察条件字段的过滤数量,如果过滤的很多,需要排序的并不多时,优先把索引放在范围字段上

FileSort算法解析

排序字段若不在索引列,则filesort会有两种算法:双路排序和单路排序

双路排序:
两次扫描磁盘,最终获得数据,读取行指针和order by列,对他们进行排序,根据排序结果再去读取对应的行数据
从磁盘中取字段,在buffer中排序,再从磁盘取其他字段
,这就会出现随机IO
单路排序:
直接读取所有列,再buffer中进行排序

FileSort调优

  • 提高sort_buffer_size,避免反复写入磁盘
  • 提高max_length_for__sort_data,增加用改进算法的概率
  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

踢足球的程序员·

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值