优秀的编程知识分享平台

网站首页 > 技术文章 正文

回表、覆盖索引(sqlserver覆盖索引)

nanyue 2025-06-10 17:01:56 技术文章 4 ℃

回表与覆盖索引对比

概念定义

回表查询

当使用非聚簇索引(二级索引)查询时,若所需字段未完全包含在索引中,需根据索引记录的主键值回到聚簇索引(主键索引)中查询完整数据行,此过程称为回表。例如:通过 username 索引找到主键 id 后,还需回主键索引获取 email 和 age 字段。

覆盖索引

索引包含查询所需的所有字段(SELECT 字段 + WHERE 条件),可直接从索引树获取结果,无需回表操作。例如:联合索引 (username, age) 覆盖查询 SELECT username, age FROM users,直接返回索引数据。

执行过程对比

场景

回表查询

覆盖索引

查询示例

SELECT email FROM users WHERE username='A'

SELECT username FROM users WHERE username='A'

索引结构

username索引仅存 username和 id

(username, email)索引包含全部查询字段

执行步骤

1. 查询 username索引获取 id
2. 回主键索引查询 email

直接通过联合索引返回 username和 email

磁盘 I/O

两次(索引树 + 聚簇索引)

一次(仅索引树)

示例说明

回表示例

-- 表结构:id(主键), username(索引), email
SELECT email FROM users WHERE username = 'John';
  • 执行过程:通过 username 索引找到 id → 回表查询主键索引获取 email。

覆盖索引示例

-- 创建覆盖索引:ALTER TABLE users ADD INDEX idx_username_email(username, email);
SELECT username, email FROM users WHERE username = 'John';
  • 优势:索引 idx_username_email 直接包含查询字段,无需回表。

优化建议

优先设计覆盖索引

  • 将高频查询的字段合并为联合索引,如 (a, b, c) 覆盖 SELECT a, b, c。避免 SELECT *,减少索引外的字段查询。

权衡索引开销

  • 覆盖索引可能增加索引体积,影响写入性能,需平衡查询效率与存储成本。

利用最左匹配原则

  • 联合索引 (a, b) 可覆盖查询 WHERE a=1 AND b=2,但无法覆盖 WHERE b=2。

性能影响对比

指标

回表查询

覆盖索引

磁盘 I/O

高(两次访问)

低(一次访问)

查询延迟

较高

较低

适用场景

查询非索引字段

查询仅含索引字段

通过合理设计索引,覆盖索引可减少 50% 以上的 I/O 开销。

最近发表
标签列表