在使用MySQL时,有时需要修改表结构或添加索引来优化性能,但这些操作是否会锁表一直是一个争议的话题。本文将从MySQL5.7和MySQL8.0两个版本的角度来探讨这个问题。
先上结论在MySQL5.7和MySQL8.0中,表结构修改和索引添加通常不会锁定整个表。但是,在某些情况下,MySQL可能需要锁定整个表。如果使用ALTERTABLE语句添加索引,则将锁定表。相反,如果使用CREATEINDEX语句添加索引,则不会锁定表。在MySQL8.0中,还引入了“InvisibleIndexes”、“InstantDDL”和“In-PlaceAlter”升级等新功能,可以进一步提高MySQL的性能和可维护性。在进行表结构修改和索引添加时,需要根据具体情况选择合适的方法,以避免对MySQL的性能产生负面影响。
MySQL5.7版本在MySQL5.7中,对于表结构修改和索引添加,MySQL使用了一种称为“In-PlaceAlter”的技术。这种技术允许在不锁定表的情况下进行修改。In-PlaceAlter使用了一种称为“FastIndexCreation”的技术来优化索引添加,它可以在不拷贝原始表数据的情况下创建新索引。
对于表结构修改,MySQL5.7使用了一种称为“OnlineDDL”的技术。OnlineDDL允许在不锁定表的情况下进行表结构修改。当修改表结构时,MySQL会创建一个新表,并将原始表中的数据复制到新表中。在复制数据期间,原始表仍然可读和可写。完成数据复制后,MySQL会将新表重命名为原始表的名称,并删除原始表。此过程通常只需要几秒钟或几分钟,并且可以在应用程序运行期间执行。
虽然In-PlaceAlter和OnlineDDL可以减少对表的锁定,但它们并不适用于所有情况。例如,在进行某些类型的表结构修改时,MySQL可能需要锁定整个表。在这种情况下,In-PlaceAlter和OnlineDDL可能不适用。
此外,在MySQL5.7中,如果使用ALTERTABLE语句添加索引,将会锁定表。但是,如果使用CREATEINDEX语句添加索引,则不会锁定表。这是因为CREATEINDEX语句使用了“ConcurrentInserts”的技术,它允许在不锁定表的情况下进行索引添加。ConcurrentInserts在索引添加期间允许其他并发操作,从而减少了锁定的需求。
综上所述,在MySQL5.7中,表结构修改和索引添加通常不会锁定整个表。但是,在某些情况下,MySQL可能需要锁定整个表。此外,如果使用ALTERTABLE语句添加索引,则会锁定表。相反,如果使用CREATEINDEX语句添加索引,则不会锁定表。
MySQL8.0版本MySQL8.0在表结构修改和索引添加方面引入了一些新的功能。其中最重要的功能是“InvisibleIndexes”和“In-PlaceAlter”升级。
“InvisibleIndexes”允许您创建不会影响查询执行计划的索引。这些索引对于实时修改和测试查询计划非常有用。此外,如果某个索引不再需要,您可以将其设置为“不可见”,而不是完全删除它。这可以减少表锁定的时间,因为在删除可见索引之前不需要锁定整个表。
“In-PlaceAlter”在MySQL8.0中进行了升级,可以处理更多类型的表结构修改。例如,现在可以在不锁定表的情况下添加、删除和更改列。如果使用In-PlaceAlter进行表结构修改,则MySQL将创建一个新的表,将新表中的数据与原始表中的数据进行同步,然后在删除原始表之前将新表重命名为原始表的名称。
在MySQL8.0中,索引添加的行为与MySQL5.7相同。如果使用ALTERTABLE语句添加索引,则将锁定表。相反,如果使用CREATEINDEX语句添加索引,则不会锁定表。使用“ConcurrentInserts”技术添加索引可以避免表锁定,这使得索引添加变得更加高效。
此外,在MySQL8.0中,还引入了“InstantDDL”功能。这种功能允许您在不锁定表的情况下执行某些DDL操作,例如添加、删除和更改列,以及更改表类型和重命名表。这可以显著提高MySQL的可用性和可维护性。
综上所述,在MySQL8.0中,表结构修改和索引添加的锁定行为与MySQL5.7大致相同。如果使用ALTERTABLE语句添加索引,则将锁定表。相反,如果使用CREATEINDEX语句添加索引,则不会锁定表。使用In-PlaceAlter进行表结构修改可以避免锁定整个表。此外,MySQL8.0引入了“InvisibleIndexes”、“InstantDDL”和“In-PlaceAlter”升级等新功能,可以进一步提高MySQL的性能和可维护性。