SQL MySQL 的 Invisible Index 与索引实验性启用/禁用实践

2次阅读

mysql 8.0+ 创建 Invisible Index 需在 CREATE INDEX 或 ALTER table 语句中添加 INVISIBLE 关键字,主键、唯一索引、全文索引和空间索引不支持该特性。

SQL MySQL 的 Invisible Index 与索引实验性启用/禁用实践

MySQL 8.0+ 怎么创建 Invisible Index

Invisible Index 是 MySQL 8.0 引入的特性,它让索引对优化器“不可见”,但物理上仍存在、仍被维护。创建时只需在 CREATE INDEXALTER TABLE 中加 INVISIBLE 关键字。

  • CREATE INDEX idx_name ON t1(col1) INVISIBLE;
  • ALTER TABLE t1 ALTER INDEX idx_name INVISIBLE;(已存在索引可直接设为 invisible)
  • 注意:主键(PRIMARY KEY)和唯一约束(UNIQUE)相关的索引不能设为 invisible —— MySQL 会报错 ER_UNSUPPORTED_INDEX_ALTER
  • 全文索引(FULLTEXT)和空间索引(SPATIAL)也不支持 invisible,尝试会触发 ER_NOT_SUPPORTED_YET

为什么 select 不走 Invisible Index 却仍慢

因为 invisible 索引只是对优化器“隐身”,不参与执行计划选择;但它依然在 INSERT/UPDATE/delete 时被更新,写放大照旧。所以如果表写入频繁,性能下降可能比预期更明显。

  • EXPLAIN format=TREE 能确认是否用了某个索引;invisible 索引不会出现在输出里
  • 但用 SHOW INDEX FROM t1 会看到 Visible 列值为 NO
  • 容易误判的点:即使你手动加了 USE INDEX(idx_name),优化器也会忽略 invisible 索引,直接报错 ER_KEY_DOES_NOT_EXITS(不是拼写错误,是“逻辑上不可用”)

如何临时“启用”一个 Invisible Index 做实验

没有运行时开关,必须显式修改索引可见性。但可以快速切回,适合 A/B 计划验证。

  • 启用:ALTER TABLE t1 ALTER INDEX idx_name VISIBLE;
  • 禁用:ALTER TABLE t1 ALTER INDEX idx_name INVISIBLE;
  • 这个操作是 instant DDL(MySQL 8.0.12+),不锁表,但要注意:它会触发一次元数据锁(MDL),如果此时有长事务正在查该表,ALTER 会等 —— 容易卡住
  • 别在高峰期批量改多个索引 visibility,每个 ALTER 都是独立 MDL 请求

哪些场景真该用 Invisible Index

它不是用来“先建着再说”的占位符,而是为索引下线提供灰度路径。

  • 想删一个疑似无用索引,但不敢直接 DROP?先 INVISIBLE,观察一周慢查询日志和 performance_schema.table_io_waits_summary_by_index_usage 中的 COUNT_STAR
  • 上线新复合索引前,把旧单列索引设为 invisible,对比 QPS 和延迟变化
  • 注意陷阱:如果应用层有 FORCE INDEX,而你把对应索引设 invisible,查询会直接失败,不是降级走全表扫描
  • 备份工具(如 mysqldump)默认不导出 invisible 索引,但物理备份(xtrabackup)会保留 —— 恢复后仍是 invisible 状态,这点常被忽略

真正难的是判断“这个索引到底有没有被隐式依赖”:比如某 ORM 自动生成 FORCE INDEX,或 dba 写的巡检脚本硬编码了索引名。invisible 不解决这类耦合,只暴露它们。

text=ZqhQzanResources