我们的数据库中有一个名为的表company_competitors
。有一项工作每天截断并加载此表。该表有两列company_id
,并且competitor_id
都引用另一个表companies
。
CREATE TABLE `company_competitors` (
`id` bigint(20) NOT NULL AUTO_INCREMENT,
`company_id` int(11) DEFAULT NULL,
`competitor_id` int(11) DEFAULT NULL,
PRIMARY KEY (`id`),
CONSTRAINT `fk_rails_company_id_c1ac450a` FOREIGN KEY (`company_id`) REFERENCES `companies` (`id`),
CONSTRAINT `fk_rails_competitor_id_772a45c6` FOREIGN KEY (`competitor_id`) REFERENCES `companies` (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=268477 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_bin
截断和加载中包含的步骤是:
- 创建一个新表
company_competitors_new
- 在两列(
company_id
,competitor_id
)上一一添加外键约束。 - 将数据加载到新表中。
- 交换表
company_competitors
和company_competitors_new
.
查询:
"CREATE TABLE company_competitors_new LIKE company_competitors;
alter table company_competitors_new ADD CONSTRAINT fk_rails_company_id_53f8f57a foreign key (company_id) references companies(id);'
在添加外键时,其他人可能会访问公司表。因此,当第二个查询运行时,数据库陷入死锁,并且没有执行对公司表的查询。
show full processlist
显示所有查询说Waiting to acquire Metadata lock
,我必须杀死所有选择查询,以便添加外键完成。
我需要帮助来理解这里死锁的原因以及我该如何处理它。我还想知道是否有更好的方法来截断和加载 0 停机时间。