如何在不加表锁的情况下在大型 MySQL 生产表上创建索引
问题背景:
在大型 MySQL 表上创建索引可能是一项艰巨的任务,尤其是在不间断访问至关重要的生产环境中。传统的 CREATE INDEX 语句可能会导致完全表锁定,从而阻塞所有并发操作。
MySQL 版本注意事项:
- 在 MySQL 5.6 及更高版本中,索引更新在线执行,允许在索引创建期间继续进行读写操作。
- 但是,在 MySQL 5.5 及更早版本中,包括 InnoDB 和 MyISAM 表,索引更新将阻止对表的写入。
循环主控方法:
对于 5.6 之前的 MySQL 版本,一种有效的方法是循环主控技术:
- 设置辅助主控( Master B) 从主 master (Master A) 复制。
- 在 Master B 上执行架构更新(允许其在升级过程中落后)。
- 确保架构更改兼容从 Master A 上的下版本架构复制命令。
- 自动将所有客户端从 Master A 切换到 Master B。
- 更新 Master A 上的架构并使其成为辅助主服务器。
Percona 的 pt-online-schema-change 工具:
此工具通过以下方式自动执行循环母版方法:
- 创建新表使用更新后的架构。
- 使用触发器使新表与原始表保持同步。
- 从原始表中批量复制行。
- 替换原始表
AWS RDS 注意事项:
对于托管在 Amazon RDS 上的 MySQL 数据库,可以使用“只读副本促销”功能来促进架构更改无需表锁定。这涉及对只读从站进行更改,然后将其提升为新的主站。
替代技术:
-
使用临时表:使用新索引创建临时表,从原表插入数据,并用临时表替换原表。
-
对表进行分区:拆分表分成较小的分区,分别在每个分区上创建索引,然后将分区合并在一起。
-
使用后台进程:创建一个单独的后台进程,在表创建时逐渐创建索引仍可进行正常操作。并非所有 MySQL 版本都支持此方法。
以上是如何在不加表锁的情况下在大型MySQL生产表上创建索引?的详细内容。更多信息请关注PHP中文网其他相关文章!