PS:现在orders表中存在idx_user_id(user_id),以及idx_user_status(user_id, status)两个索引
1-例子
假设orders表内部有10w条数据,满足user_id = 1 AND status = 1的有5w条,运行:
mysql
SELECT user_id, status, amount FROM orders WHERE user_id = 1 AND status = 1;
此时会用到联合索引idx_user_status(user_id, status),但是我们还查询了一个字段amount,此时,就要去回表了,就是用现在的结果,再通过id去拿amount,流程如下:
bash
步骤1: 在二级索引 idx_user_status 中找到 50,000 个匹配项
↓
步骤2: 对每一条记录,拿着主键 id 去回表查 amount
↓
├─> 回表 1 次:根据 id=101 去聚簇索引查 amount
├─> 回表 2 次:根据 id=205 去聚簇索引查 amount
├─> ... 重复 50,000 次 ...
└─> 回表 50,000 次:根据 id=xxxxx 查 amount
↓
步骤3: 返回结果
PS:针对高频查询,尽量让 WHERE 条件 + SELECT 字段 都被同一个索引覆盖,彻底消灭回表
2-回表
用现在的结果,再通过id去拿其他信息
3-回表的方式
- MySQL的InnoDB存储引擎里,二级索引(比如你建的联合索引)存储的是索引字段 + 主键值
- 当你需要额外字段(如
amount),数据库会用索引定位到主键,然后再通过主键去表里查对应的行 - 所以确实是每一条匹配的记录都要回表一次,在上方的例子里就是5万次回表