IMO,下面的mysql不会点击索引当add_time字段用date() func包装时,但是从解释结果来看,它击中了索引,为什么?
explain
SELECT count(0)
FROM xxx
WHERE date(add_time) >= date_sub(curdate(), INTERVAL 7 day);

发布于 2019-04-22 15:03:28
众所周知,解释报告很难解释。他们在几个字段中超载了太多的信息。你可以试试
type: index表示它正在进行索引扫描,这意味着它正在访问索引中的每个条目。
这将访问与表扫描相同的条目数,但它针对的是次要索引,而不是聚集(主)索引。
当我们看到type: index时,解释显示possible_keys: NULL,这意味着它不能使用任何索引来有效地搜索。但是它也显示了key: add_time,这意味着它用于索引扫描的索引是add_time。
索引扫描是由于MySQL本身无法优化表达式或函数调用。例如,如果要搜索具有特定月份的日期,可以搜索month(add_time) = 4,但这不会在add_time上使用索引,因为该月份的日期分散在索引中,并不是所有的日期都分组在一起。
您可能知道date(add_time)应该能够通过索引进行搜索,但是MySQL并没有这样的推断。MySQL只是看到您使用的是一个函数,它甚至没有尝试使用索引。
这就是为什么MySQL 5.7引入了生成列以允许对表达式进行索引,而MySQL 8.0通过允许为表达式定义的索引 (而不需要首先定义生成的列)使它更好。
发布于 2019-04-22 13:59:41
我想你可能误解了这里的解释计划。如果您在possible_keys下面查看,您将看到它是NULL。来自解释输出格式的MySQL 文档:
如果该列为NULL (或在JSON格式的输出中未定义),则不存在相关索引。
因此,在Extra部分中提到的索引可能与add_time无关。
https://stackoverflow.com/questions/55795485
复制相似问题