Mysql实例分析一个MySQL的异常查询的案例
《Mysql实例分析一个MySQL的异常查询的案例》要点: 问题MYSQL教程 用户工单疑问:相同的语句,只是最后的limit行数不同.奇怪的是,limit 10 的性能比limit 100的语句还慢约10倍.MYSQL教程 隐藏用户表信息,语句及结果如下 SELECT f1,SUM(`f2`) `CNT` FROM T WHERE f1 IS NOT NULL AND f3 = '2014-05-12' GROUP BY f1 ORDER BY `CNT` DESC LIMIT 10; 执行时间3 min 3.65 secMYSQL教程 SELECT f1,SUM(`f2`) `CNT` FROM T WHERE f1 IS NOT NULL AND f3 = '2014-05-12' GROUP BY f1 ORDER BY `CNT` DESC LIMIT 100; 执行时间1.24Sec.MYSQL教程 性能差距非常大!MYSQL教程 分析 更有冲击性的效果是通过缩小范围后,在这个数据下,limit 67和limit 68的执行计划相差很大.MYSQL教程 两个执行计划: LIMIT 67 id: 1 select_type: SIMPLE table: a type: range possible_keys: A,B,C key: B key_len: 387 ref: NULL rows: 2555192 Extra: Using where; Using temporary; Using filesort 1 row in set (0.00 sec) LIMIT 68 id: 1 select_type: SIMPLE table: a type: ref possible_keys: A,C key: A key_len: 3 ref: const rows: 67586 Extra: Using where; Using temporary; Using filesort 1 row in set (0.00 sec) 可以看到,两个语句的执行计划不同:使用的索引不同.MYSQL教程 MySQL Tips:explain的结果中,key表示最终使用的索引,rows表示使用这个索引需要扫描的行数,这是个估计值.MYSQL教程 表中 索引A定义为 (f3,f4,f1,f2,f5); 索引B定义为(f1,f3);MYSQL教程 一个确认MYSQL教程 虽然rows是估计值,但是指导索引使用的依据.既然limit 68能达到rows 67586,说明在第一个语句优化器可选结果中,也应该有此值,为什么不会选择索引A? MySQL Tips:MySQL语法中能够用force index 来强行要求优化器使用某一个索引.MYSQL教程 Explain SELECT f1,SUM(f2) CNT FROM t force index(A) WHERE f1 IS NOT NULL AND f3 = ‘2014-05-12' GROUP BY P ORDER BY CNT DESC LIMIT 67G id: 1 select_type: SIMPLE table: a type: ref possible_keys:A key: A key_len: 3 ref: const rows: 67586 Extra: Using where; Using temporary; Using filesort 1 row in set (0.00 sec) 顺便说明,由于我们指定了force index,因此优化器不会考虑其他索引,possible_keys里只会显示A.我们关注的是rows:67586.这说明在limit 67语句里,使用索引A也能够减少行扫描.MYSQL教程 MySQL Tips:MySQL优化器会对possiable_key中的每个可能索引都计算查询代价,选择最小代价的查询计划.MYSQL教程 至此我们大概可以猜测,这个应该是MySQL实现上的bug:没有选择合适的索引,导致使用了明显错误的执行计划.MYSQL教程 MySQL Tips:MySQL的优化器执行期间需要依赖于表的统计信息,而统计信息是估算值,因此有可能导致得到的执行计划非最优.MYSQL教程 但要说明的是,上述Tip是客观情况造成(可接受),但本例却是例外,因此优化器实际上可以拿到能够作出选择正确结果的数据(rows值),但是最终选择错误.MYSQL教程 原因分析MYSQL教程 MySQL优化器是按照查询代价的估算值,来确定要使用的索引.计算这个估算值的过程,基本是按照“估计需要扫描的行数”来确定的.MYSQL教程 MySQL Tips:MySQL在目前集团主流使用的5.1和5.5版本中只能使用前缀索引.MYSQL教程 因此,使用索引A只能用上字段f3,使用索引B只能用上字段f1.Rows即为使用了索引查到上下界,之后需要扫描的数据行数(估算值).MYSQL教程 上述的语句需要用到group和order by,因此执行计划中都有Using temporary; Using filesort. 之后依次计算其他possitabe_key的查询代价.由于过程中需要排序,在得到一个暂定结果后,需要判断是否有代价更低的排序方式(test_if_cheaper_ordering). 在这个逻辑的实现过程中,存在一个bug:在估计当前索引的区分度的时候,没有考虑到前缀索引.MYSQL教程 即:假设表中有50w行数据,索引B(f1,f3),则计算索引区分度时,需要根据能够用上的前缀部分来确定.比如f1有1000个不同的值,则平均每个key值上的记录数为500.如(f1,f2)有10000个同的值,则平均每个组合key上的记录数为50,若(f1,f3)有50w个不同的值,则平均每个组合key上的记录数为1.MYSQL教程 MySQL Tips:每个key上的记录数越少,说明使用该索引查询时效率最高.对应于show index from tbl 输出结果中的Cardinality值越大.MYSQL教程 在这个case下,索引B只能使用f1做前缀索引,但是在计算单key上的行平均值时用的是(f1,这就导致估算用索引B估算的时候,得到的代价偏小.导致误选.MYSQL教程 回到问题本身MYSQL教程 1、 为什么limit值大的时候反而选对了呢? 2、 这个表有50w行数就,为什么limit相差为就差别这么大? 解决方案MYSQL教程 分析清楚后解决方法就比较简单了,修改代码逻辑,在执行test_if_cheaper_ordering过程中,改用字段f1的区分度来计算即可. (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |