主索引与索引的查询性能

Query performance on primary index vs index

我在 mysql 上有一个 table 和两个性能完全不同的查询。我提取了查询计划,但我无法完全理解性能差异背后的原因。

table:

+-------------+----------------------------------------------+------------------------------------+
|   TableA    |                                              |                                    |
+-------------+----------------------------------------------+------------------------------------+
| id          | int(10)   unsigned NOT NULL AUTO_INCREMENT   |                                    |
| userId      | int(10)                                      | unsigned DEFAULT NULL              |
| created     | timestamp                                    | NOT NULL DEFAULT CURRENT_TIMESTAMP |
| PRIMARY KEY | id                                           |                                    |
| KEY userId  | userId                                       |                                    |
| KEY created | created                                      |                                    |
+-------------+----------------------------------------------+------------------------------------+

Keys/Indices:id字段上的主键,userId字段ASC上的一个键 , created 字段 ASC 上的另一个键。

tableA 是一个非常大的 table,它包含数百万行。

我运行在这个table上的查询是:

ID为1234的用户在此table中有150万条记录。我想获取其最新的 100 行。为了实现这一点,我有 2 个不同的查询:

查询 1:

SELECT * FROM tableA USE INDEX (userId) 
WHERE userId=1234 ORDER BY created DESC LIMIT 100;

查询 2:

SELECT * FROM tableA 
WHERE userId=1234 ORDER BY id DESC LIMIT 100;

由于tableAid字段是自增的,所以保留最新的条件。这 2 个查询 return 相同的结果。但是,存在巨大的性能差异。

查询计划是:

+----------+-----------------------------------------------+-------------------------------+------+---------------------------------------+
| Query No |                   Operation                   |            Params             | Raws |               Raw desc                |
+----------+-----------------------------------------------+-------------------------------+------+---------------------------------------+
| Query 1  | Sort(using file sort) Unique index scan (ref) | table: tableA; index: userId; | 2.5M | Using index condition; Using filesort |
| Query 2  | Unique index scan (ref)                       | table: tableA; index: userId; | 2.5M | Using where                           |
+----------+-----------------------------------------------+-------------------------------+------+---------------------------------------+


+--------+-------------+
|        | Performance |
+--------+-------------+
| Query1 | 7,5 s       |
+--------+-------------+
| Query2 | 741 ms      |
+--------+-------------+

我了解到查询1有一个排序操作。在每个查询中,使用的索引是userId。但是为什么查询 2 中没有使用排序呢?主索引如何影响?

Mysql 5.7

编辑:table 上还有更多列,我从上面的 table 定义中提取了它们。

Since id field of tableA is auto increment, the condition of being latest is preserved.

那是通常一个有效的陈述。

WHERE userId=1234 ORDER BY created DESC LIMIT 100

需要这个 'composite' 索引:(userId, created)。这样一来,无论 table 大小或该用户的行数如何,它只会命中 100 行。

同样适用于

WHERE userId=1234 ORDER BY id DESC LIMIT 100;

即需要(userId, id)。但是,在 InnoDB 中,当您说 INDEX(x) 时,它会默默地添加到 PRIMARY KEY 列。所以你有效地得到 INDEX(x,id)。这就是为什么您的普通 INDEX(userId) 效果很好。

EXPLAIN 很少(如果有的话)考虑 LIMIT。这就是为什么 'Rows' 对于两个查询都是“2.5M”。

如果您去掉 USE INDEX 提示,第一个查询可能(或可能不会)使用 INDEX(userId)。选择取决于 table 中 userId = 1234 的百分比。如果它小于大约 20%,将使用该索引。但它会在二级索引和数据之间来回跳动——总共 150 万次。如果超过 20%,它将通过简单地读取所有 "millions" 行来避免弹跳,忽略那些不适用的行。

注意:您为 Q1 所做的仍然会读取至少 1.5M 行,对它们进行排序 ("Using filesort"),然后剥离所需的 100。但是对于 INDEX(userId, created),它可以跳过排序并仅查看 100 行。

没有看到 SHOW CREATE TABLE 和 un-annotated EXPLAIN,我无法解释 "Unique index scan"。 (EXPLAIN FORMAT=JSON SELECT... 可能会提供更多见解。)