MySQL 8.0+ 并行创建索引默认关闭,由innodb_ddl_threads(8.0.27+)或innodb_sort_buffer_size与线程调度共同决定;二级索引可并行,主键重建仍单线程。

MySQL 8.0+ 并行创建索引到底开不开?
默认不开,innodb_parallel_read_threads 控制的是并行读线程数,和建索引无关;真正影响 DDL 并行度的是 innodb_ddl_threads(MySQL 8.0.27+)或更底层的 innodb_sort_buffer_size + 线程调度。老版本(innodb_sort_buffer_size 足够大。
- MySQL 8.0.27+:设
innodb_ddl_threads > 1才可能触发多线程索引构建(仅对二级索引生效,主键重建仍单线程) -
innodb_sort_buffer_size建议调到 4M–16M(太小导致频繁落盘,太大引发内存争抢) - 并行效果高度依赖数据分布:若
WHERE条件筛选率低、或索引字段重复值极高,并行归并反而变慢
ALTER TABLE ADD INDEX 卡住不动?先看这三件事
不是卡,是正在做隐式拷贝或 Online DDL 的 prepare 阶段——尤其在大表上,连 SHOW PROCESSLIST 都只显示 altering table,毫无进度提示。根本原因常是锁等待、I/O 饱和或 buffer pool 污染。
- 检查是否被其他长事务阻塞:
SELECT * FROM performance_schema.data_locks+data_lock_waits - 确认磁盘 I/O 是否打满:
iostat -x 1看%util和await,SSD 上await > 20ms就算异常 - 临时关闭
innodb_change_buffering = 'none',避免 change buffer 合并拖慢建索引过程(尤其机械盘)
为什么加完索引查询还是慢?索引没生效的隐蔽原因
建索引成功 ≠ 查询走索引。大表上常见「索引建了,EXPLAIN 还是 type: ALL」,问题往往不在 SQL 写法,而在统计信息陈旧或索引选择性崩坏。
- 执行
ANALYZE TABLE强制更新统计信息(InnoDB 默认采样率低,大表易误判) - 用
SELECT COUNT(DISTINCT col)/COUNT(*) FROM tbl算选择性:低于 0.01 时优化器大概率放弃该索引 - 注意隐式类型转换:
WHERE varchar_col = 123会触发全表扫描,哪怕varchar_col有索引
在线建索引期间写入抖动大?这是正常现象
Online DDL 不是零代价,它靠「row log」记录变更,在 build index 阶段回放。当并发写入高、索引字段更新频繁时,row log 会快速膨胀,触发 flush 和 merge,直接冲击 buffer pool 和 redo log。
- 监控
Innodb_dblwr_writes和Innodb_log_waits:后者持续 > 0 说明 redo log 太小或刷盘跟不上 - 建索引前临时调大
innodb_log_file_size(需重启)或innodb_log_buffer_size(动态) - 避开业务高峰,或用
pt-online-schema-change分摊压力——但它会额外增加主从延迟和磁盘空间占用
大表建索引从来不是“配个参数就快起来”的事,真正耗时的永远是磁盘寻道、buffer pool 淘汰策略、以及优化器对统计信息的误判——这些环节没有银弹,只有针对性地查、调、等。










