我有一个名为CacheSync的模型,mysql显示它有一个索引:
mysql> show indexes from cache_syncs; +-------------+------------+---------------------------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+ | Table | Non_unique | Key_name | Seq_in_index | Column_name | Collation | Cardinality | Sub_part | Packed | Null | Index_type | Comment | Index_comment | +-------------+------------+---------------------------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+ | cache_syncs | 0 | PRIMARY | 1 | id | A | 90878 | NULL | NULL | | BTREE | | | | cache_syncs | 1 | index_cache_syncs_on_created_at | 1 | created_at | A | 18175 | NULL | NULL | YES | BTREE | | | +-------------+------------+---------------------------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+ 2 rows in set (0.01 sec)
但是当我去解释时,它说它没有使用索引:
CacheSync.where("created_at < ?", (Time.now - 1.hour).to_time).explain => EXPLAIN for: SELECT `cache_syncs`.* FROM `cache_syncs` WHERE (created_at < '2022-06-13 19:37:23.316439') +----+-------------+-------------+------+---------------------------------+------+---------+------+-------+-------------+ | id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra | +----+-------------+-------------+------+---------------------------------+------+---------+------+-------+-------------+ | 1 | SIMPLE | cache_syncs | ALL | index_cache_syncs_on_created_at | NULL | NULL | NULL | 93651 | Using where | +----+-------------+-------------+------+---------------------------------+------+---------+------+-------+-------------+ 1 row in set (0.00 sec)
为什么不使用索引?
感谢您的帮助, 凯文
P粉4211197782024-03-22 10:04:01
根据我的经验,如果优化器估计您的条件与表的 20% 以上匹配,它将回退到表扫描。它猜测从聚集索引中读取所有行比在二级索引中查找值更快,然后再进行一次查找以从表中获取相应的行。
20% 的阈值不是任何官方功能,这只是我观察到的。它在当前版本的 MySQL 中不可配置。
您可以使用索引提示来说服它表扫描的成本过高:
SELECT ... FROM mytable FORCE INDEX (index_cache_syncs_on_created_at) WHERE ...
只有当您指定的索引与查询中的条件无关时,它才会执行表扫描。
参见https://dev.mysql.com/ doc/refman/8.0/en/index-hints.html 有关索引提示的更多信息。
我不是 Rails 开发人员,但这个旧答案显示了一种向 Rails 传递索引提示语法的方法:https:// stackoverflow.com/a/13904227/20860 我不知道这是否仍然是目前的做法。