2

我创建了一个 MySQL 表,它按日期列的月份进行分区,如下所示:

CREATE TABLE `my_big_table` (
  `some_id` varchar(60) COLLATE utf8_unicode_ci NOT NULL,
  `some_value` varchar(256) COLLATE utf8_unicode_ci NOT NULL,
  `some_other_value` varchar(256) COLLATE utf8_unicode_ci NOT NULL,
  `time` datetime NOT NULL
) ENGINE=MyISAM DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci
PARTITION BY LIST(MONTH(time))
(PARTITION p1 VALUES IN (1) ENGINE = MyISAM,
 PARTITION p2 VALUES IN (2) ENGINE = MyISAM,
 PARTITION p3 VALUES IN (3) ENGINE = MyISAM,
 PARTITION p4 VALUES IN (4) ENGINE = MyISAM,
 PARTITION p5 VALUES IN (5) ENGINE = MyISAM,
 PARTITION p6 VALUES IN (6) ENGINE = MyISAM,
 PARTITION p7 VALUES IN (7) ENGINE = MyISAM,
 PARTITION p8 VALUES IN (8) ENGINE = MyISAM,
 PARTITION p9 VALUES IN (9) ENGINE = MyISAM,
 PARTITION p10 VALUES IN (10) ENGINE = MyISAM,
 PARTITION p11 VALUES IN (11) ENGINE = MyISAM,
 PARTITION p12 VALUES IN (12) ENGINE = MyISAM)

现在,当我跑步时

explain partitions select * from my_big_table where month(time) = 2 limit 10;

我得到输出:

id  select_type table       partitions              type    possible_keys   key key_len ref rows        Extra
1   SIMPLE      my_big_table    p1,p2,p3,p4,p5,p6,p7,p8,p9,p10,p11,p12  ALL NULL        NULL    NULL    NULL    10762868    "Using where"

所以显然所有的分区都被读取了,我想知道为什么。引擎不应该足够聪明地找出匹配值只能在分区p2中找到吗?对于基于日期的查询,我得到相同的输出,例如

explain partitions select * from my_big_table where date(time) = '2013-02-01' limit 10;

有什么线索吗?

干杯,亚历克斯

4

3 回答 3

2

除此以外的所有口味PARTITION BY RANGE都是无用的。当您有多个值(如您所见)时尤其如此。

相反,考虑PARTITION BY RANGE(TO_DAYS(time))和使用WHERE time >= '2014-02-01' AND time < '2014-02-01' + INTERVAL 7 MONTH(以扫描一个月为例)。不要使用. _MONTH(time)

更多讨论

于 2015-11-26T04:51:43.633 回答
1

好的,经过更多的谷歌搜索和更多
的试验和错误:问题似乎与我查询的 where 子句中的月份(...)或日期(...)语句有关。查询

explain partitions select * from my_big_table where time = '2013-02-01 00:00:00' limit 10;

仅按预期扫描分区p2 。因此,似乎分区修剪仅在分区列的原始值是 where 子句的一部分时才有效。

干杯,亚历克斯

于 2013-04-02T16:18:13.343 回答
0

你永远不应该对表值使用函数month(time) = 2,MySQL 将始终调查此类查询的所有值(即全表扫描)。

通常,您会发现这种与未使用索引相关的“问题”,但分区修剪也是如此。

于 2014-04-30T18:49:03.033 回答