1

我有下一张桌子:

CREATE TABLE `test` (
 `fingerprint` varchar(80) COLLATE utf8_unicode_ci NOT NULL,
 `country` varchar(5) COLLATE utf8_unicode_ci NOT NULL,
 `loader` int(10) unsigned NOT NULL,
 `date` date NOT NULL,
 `installer` int(10) unsigned DEFAULT NULL,
 `browser` varchar(5) COLLATE utf8_unicode_ci NOT NULL DEFAULT '',
 `version` varchar(5) COLLATE utf8_unicode_ci NOT NULL DEFAULT '',
 `os` varchar(10) COLLATE utf8_unicode_ci NOT NULL DEFAULT '',
 `language` varchar(10) COLLATE utf8_unicode_ci NOT NULL DEFAULT '',
 PRIMARY KEY (`fingerprint`, `date`),
 KEY `date_1` (`date`),
 KEY `date_2` (`date`,`loader`,`installer`,`country`,`browser`,`os`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;

现在它包含 1000 万条记录,并且每 200 万条记录/天会增加。

我的问题,为什么 MySQL 在下一个查询中使用“Using Where”:

explain select count(*) from test where date between '2013-08-01' and '2013-08-10'
1   SIMPLE  test    range   date_1,date_2   date_1  3       1601644 Using where; Using index

更新,为什么下一个问题有类型 - All 和 Using where then:

explain select * from test use key(date_1) where date between '2013-08-01' and '2013-08-10'
1 SIMPLE test ALL date_1 null null null 3648813 Using where
4

2 回答 2

1

确实使用索引。

它在那里说:Using where; Using index。“使用位置”并不意味着完全扫描,它意味着它正在使用WHERE您提供的条件。

1601644 数字也暗示了这一点:这意味着它预计读取大约 160 万条记录,而不是表中的全部 1000 万条记录,并且它与您的 ~200 万/天估计相关。

简而言之,它似乎做得很好,只是您将检索到很多数据。

尽管如此,它也在读取表数据,当索引似乎足够时。尝试更改count(*)with count(date)date整个查询中提到的唯一字段也是如此。如果你得到 only Using index,那么它可能会更快。

于 2013-08-07T13:30:47.573 回答
0

您的查询不仅仅是“使用 where”,它实际上是“使用 where;使用索引”。这意味着索引用于匹配您的WHERE条件,并且索引用于执行键值的查找。这是最好的情况,因为实际上该表从未被扫描过,查询只能使用索引来处理。

在这里,您可以找到您正在查看的输出含义的完整描述。


您的第二个查询仅显示“使用位置”通知。这意味着索引仅用于过滤行。必须从表中读取数据(没有“使用索引”通知),因为索引不包含所有行数据(您选择了所有列,但选择的索引仅涵盖date)。如果您有一个覆盖索引(涵盖所有列),则可能会使用此索引。

于 2013-08-07T13:31:45.280 回答