数据分布对MySQL执行计划的影响

[复制链接]
查看11 | 回复5 | 2010-10-8 09:31:22 | 显示全部楼层 |阅读模式
以前我一直以为,MySQL优化器只会根据数据的基数来判断执行计划的优化,在5.0时,基本上只要基数达不到要求,MySQL就不会选择索引。不过从今天优化一组SQL的情况来看,5.1早已不是这样,MySQL优化器考虑了数据分布的影响,使用不同的值,对同一条SQL可能产生完全不同的执行计划。可以做如下测试。
有一张表 A (id,c1),假设采用“SELECT c1,count(*) FROM table GROUP BY c1”的方式来查看每一列每个值的数据量的结果如下:
c1
count(*)
1
100
2
1000
3
10000
4
100000
假设现在c1上有个索引 idx_1 (c1)。
当我采用只有100个值的1作为条件的筛选值查看执行计划时,
MySQL会毫不犹豫的走 idx_1 索引。
但是改为4作为筛选值,基本上都可以看到全表扫描的执行计划。
EXPLAIN SELECT * FROM A WHERE c1=1;
EXPLAIN SELECT * FROM A WHERE c1=4;
虽然从源码目录的 sql/sql_select.cc 中并未找到具体的优化代码,但是从sql/opt_range.cc来看,至少RANGE查询的优化已经通过蒙特卡罗方法来估算要选择的值的出现概率,那么有理由相信,如此简单的选择查询,MYSQL肯定也估算了要查询的值出现的概率,然后以此为依据计算访问路径。
所以,一条SQL优化时,不仅要看数据的离散度,还得看经常被当作筛选值的数值唯一性是否够好,如果被筛选的值唯一性不好,建立索引依然是没有什么意义的,因为MySQL根本不会选择这个索引。
回复

使用道具 举报

千问 | 2010-10-8 09:31:22 | 显示全部楼层
mysql也弄个柱状图就不错了!
回复

使用道具 举报

千问 | 2010-10-8 09:31:22 | 显示全部楼层
蒙特卡罗方法来估算要选择的值的出现概率
回复

使用道具 举报

千问 | 2010-10-8 09:31:22 | 显示全部楼层
从某种程序上来看,这也是一种好事啊。
如果c1=4也选择走索引,可能还没有全表扫描来得快。
MYSQL更智能。哈
回复

使用道具 举报

千问 | 2010-10-8 09:31:22 | 显示全部楼层
长见识了
回复

使用道具 举报

千问 | 2010-10-8 09:31:22 | 显示全部楼层
看来mysql越来越合理了,做的越来越好了,大家以后多多支持,多多支持吧!
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

主题

0

回帖

4882万

积分

论坛元老

Rank: 8Rank: 8

积分
48824836
热门排行