一次数据清理引发的教训:缺失外键索引导致 DELETE 性能问题
一次数据清理引发的教训:缺失外键索引导致 DELETE 性能问题 最近在做一次数据清理(data purge)时,我遇到了一个之前没有预料到的问题。 从业务逻辑上看,这只是一个很普通的 DELETE 操作,根据条件删除历史数据而已。 但实际执行时, DELETE 语句几乎卡住了 。 没有报错,看起来也在运行,但迟迟没有完成。 问题出在哪里? 在 DBA 的帮助下,我们开始从数据库层面分析这个问题。 最终发现,真正的原因是: 部分外键(Foreign Key)字段缺少对应的索引。 在 Oracle 中,当对父表执行 DELETE 操作时,数据库需要检查子表中是否存在 引用该记录的外键数据。 如果外键列没有索引,Oracle 就只能对子表做全表扫描。 当数据量变大时,这种全表扫描会极大地拖慢 DELETE 操作, 甚至导致长时间锁表,看起来就像 SQL “卡住”了一样。 解决方案 在确认问题后,我们为所有缺失的外键字段补充了索引(涉及多个关联表)。 效果非常明显: DELETE 操作执行速度大幅提升 锁定时间明显缩短 数据库整体负载显著下降 SQL 没变,数据没变,只是补了索引,结果却完全不同。 如何定位缺失的外键索引 DBA 使用下面的 SQL 来检查哪些外键约束没有对应的索引。 以下示例中,schema 名、表名和索引名均已做匿名处理。 WITH fk_constraints AS ( SELECT c.table_name, c.constraint_name AS fk_name, LISTAGG(c.column_name, ', ') WITHIN GROUP (ORDER BY c.position) AS fk_columns FROM dba_cons_columns c JOIN dba_constraints k ON k.constraint_name = c.constraint_name WHERE k...